C++linux高并发服务器项目实践 day8

news2025/1/11 14:46:09

C++linux高并发服务器项目实践 day8

  • 内存映射
    • 内存映射相关系统调用
    • 例子
    • 思考问题
    • 案例2
    • 案例3
  • 信号
    • 信号的5中默认处理动作
    • 信号相关函数
      • kill、raise和abort函数
      • alarm函数
        • 案例
      • setitimer函数
    • signal信号捕捉函数

内存映射

内存映射是将磁盘文件的数据映射到内存,用户通过修改内存就能修改磁盘文件
在这里插入图片描述

内存映射相关系统调用

#include <sys/mman.h>
void *mmap(void *addr,size_t length,int prot,int flags,int fd,off_t offset);
int munmap(void *addr ,size_t length);

#include <sys/mman.h>
void *mmap(void *addr, size_t length, int prot, int flags,int fd, off_t offset);

  • 功能:将一个文件或者设备的数据映射到内存中
  • 参数:
    • void *addr:映射内存的首地址,NULL,由内核指定
    • length:要映射的数据的长度这个值不能为0.建议使用文件的长度
      获取文件的长度:sta lseek
    • prot:对申请的内存映射去的操作权限
      • PROT_EXEC:可执行的权限
      • PROT_READ:可读的权限
      • PROT_WRITE:写权限
      • PROT_NONE:没有权限
        要操作映射内存,必须要有读的权限。
        一般是PROT_READ或者PROT_READ|PROT_WRITE
    • flags:
      • MAP_SHARED : 映射群的数据会自动和磁盘文件进行同步,进程间通信,必须要设置这个选项
      • MAP_PRIVATE: 不同步,内存映射去的数据改变了,对原来的文件不会修改,会重新创建一个新的文件(copy on write)
    • fd:需要映射的文件的文件描述符
      • 通过open得到,open的是一个磁盘文件
      • 注意:文件的大小不能为0、open指定的权限不能和prot参数有冲突
        prot:PROT_READ open:只读/读写
        prot:PROT_READ | PROT_WRITE open:读写
    • offset:偏移量,一般不用。必须指定的是4k的整数倍,0表示不偏移
  • 返回值:返回创建的内存的首地址
    失败返回MAP_FAILED,(void *)-1

int munmap(void *addr, size_t length);

  • 功能:释放内存映射
  • 参数:
    • addr : 要释放的内存的首地址
    • length : 要释放的内存的大小,要和mmap函数中的length参数的值一样

例子

使用内存映射实现进程间通信:

  1. 有关系的进程(父子进程)
    • 还没有子进程的时候
      • 通过唯一的父进程,先创建内存映射区
    • 有了内存映射区以后,创建子进程
    • 父子进程共享创建内存映射区
  2. 没有关系的进程间通信
    • 准备一个大小不是0的磁盘文件
    • 进程1 通过磁盘文件创建内存映射区
      • 得到一个操作这块内存的指针
    • 进程2 通过磁盘文件创建内存映射区
      • 得到一个操作这块内存的指针
    • 使用内存映射区通信

注意:内存映射区通信,是非阻塞

#include <stdio.h>
#include <sys/mman.h>
#include <fcntl.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <wait.h>

//作业:使用内存映射实现没有关系的进程间的通信
int main(){

    //1. 打开一个文件
    int fd =  open("test.txt",O_RDWR);
    if(fd == -1){
        perror("open");
        return -1;
    }

    int size = lseek(fd,0,SEEK_END); //获取文件的大小

    //2.创建内存映射
    void *ptr = mmap(NULL,size,PROT_READ | PROT_WRITE,MAP_SHARED,fd,0);
    if(ptr == MAP_FAILED){
        perror("mmap");
        exit(0);
    }

    //3.创建子进程
    pid_t pid = fork();
    if(pid > 0){
        wait(NULL);
        //父进程
        strcpy((char*)ptr , "hello!Mike!!");
    }else if(pid == 0){
        //子进程
        char buf[256];
        strcpy(buf,(char *)ptr);
        printf("read data:%s\n",buf);
    }

    //关闭内存映射区
    munmap(ptr,size);

    close(fd);

    return 0;
}

思考问题

如果对mmap的返回值(ptr)做++操作(ptr++),munmap是否能够成功?

void * ptr = mmap(…);
ptr++;//可以对其进行++操作
munmap(ptr,len); //错误,不能正确释放,要保存一下地址,传入保存的地址是可行的

如果open时O_RDONLY,mmap时prot参数指定PROT_READ | PROT_WRITE会怎样?

错误,返回MAP_FAILED
open()函数中的权限建议和prot参数的权限保持一致

如果文件偏移量为1000会怎样?

偏移量必须是4k的整数倍,错误的话会返回MAP_FAILED

mmap什么情况下会调用失败?

  1. 第二个参数:length = 0
  2. 第三个参数:prot 只指定了写权限
  3. prot PROT_READ | PROT_WRITE 第5个参数fd 通过open函数时指定的O_RDONLY | O_WRONLY

可以open的时候O_CREAT一个新文件来创建映射区吗?

可以的,但是创建的文件的大小如果为0的话,肯定不行
可以对新的文件进行扩展

  • lseek()
  • truncate()

mmap后关闭文件描述符,对mmap映射有没有影响?

int fd = open(“XXX”);
mmap(,fd,0);
close(fd):
映射区还存在啊,创建映射区的fd被关闭,没有任何影响

对ptr越界操作会怎样?

void * ptr = mmap(NULL,100,);
会以内存分页的大小指定内存,4k的倍数
越界操作操作的是非法的内存 --》段错误

案例2

需求:使用内存映射实现文件拷贝的功能

思路:

  1. 对原始的文件进行内存映射
  2. 创建一个新文件(拓展该文件)
  3. 把新文件的数据映射到内存中
  4. 通过内存拷贝将第一个文件的内存数据拷贝到新的文件内存中
  5. 释放资源
#include <stdio.h>
#include <sys/mman.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>

int main(){
    //1. 对原始的文件进行内存映射
    int fd = open("english.txt",O_RDWR);
    if(fd == -1){
        perror("open");
        exit(0);
    }

    //获取原始文件的大小
    int len = lseek(fd,0,SEEK_END);

    //2. 创建一个新文件(拓展该文件)
    int fd1 = open("cpy.txt",O_RDWR|O_CREAT,0664);
    if(fd1 == -1){
        perror("open");
        exit(0);
    }

    //拓展文件
    truncate("cpy.txt",len);
    write(fd1," ",1);

    //3. 把新文件的数据映射到内存中
    void *ptr = mmap(NULL,len,PROT_READ|PROT_WRITE,MAP_SHARED,fd,0);
    void *ptr1 = mmap(NULL,len,PROT_READ|PROT_WRITE,MAP_SHARED,fd1,0);

    if(ptr == MAP_FAILED){
        perror("mmap");
        exit(0);
    }
    if(ptr1 == MAP_FAILED){
        perror("mmap");
        exit(0);
    }

    //内存拷贝
    memcpy(ptr1,ptr,len);

    //释放资源
    munmap(ptr1,len);
    munmap(ptr,len);
    //后打开先释放,先打开后释放

    close(fd);
    close(fd1);

    return 0;
}

案例3

匿名映射 :不需要文件实体进行内存映射

#include <stdio.h>
#include <sys/mman.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/wait.h>

int main(){

    //1. 创建匿名内存映射区
    int len = 4096;
    void * ptr = mmap (NULL,len,PROT_READ | PROT_WRITE,MAP_SHARED|MAP_ANONYMOUS,-1,0);
    if(ptr == MAP_FAILED){
        perror("mmap");
        exit(0);
    }

    //父子进程间通信
    pid_t pid = fork();

    if(pid > 0){
        //父进程
        strcpy((char *) ptr,"hello,world");
        wait(NULL);
    }else if(pid == 0){
        //子进程
        sleep(1);
        printf("%s\n",(char*)ptr);
    }

    //释放内存映射区 
    int ret = munmap(ptr,len);
    if(ret == -1){
        perror("munmap");
        exit(0);
    }

    return 0;
}

信号

信号是Linux进程间通信的最古老的方式之一,是时间发生时对进程的通知机制,有时也称之为软件中断,它是在软件层次上对中断机制的一种模拟,是一种异步通信的方式。信号可以导致一个正在运行的进程被另一个正在运行的异步进程中断,转而处理某一个突发事件。

发往进程的诸多信号,通常都是源自于内核。引发内核为进程产生信号的各类事件如下:

  • 对于前台进程,用户可以通过输入特殊的终端字符来给它发送信号。比如输入ctrl+c通常会给进程发送一个中断信号
  • 硬件发生异常,即硬件检测到一个错误条件并通知内核,随即再由内核发送相应信号给相关进程。比如执行一条异常的机器语言指令,诸如被0除,或者引用了无法访问的内存区域
  • 系统状态变化,比如alarm定时器到期将引起SIGALRM信号,进程执行的CPU时间超限,或者该进程的某个子进程退出
  • 运行kill命令或调用kill函数

使用信号的两个主要目的是:

  • 让进程知道已经发生了一个特定事件
  • 强迫进程执行他自己代码中的信号处理程序

信号的特点:

  • 简单
  • 不能携带大量信息
  • 满足某个特定条件才发送
  • 优先级比较高

查看系统定义的信号列表: kill -l
前31个信号为常规信号,其余为实时信号

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

信号的5中默认处理动作

  • 查看信号的详细信息:man 7 signal
  • 信号的5种默认处理动作:
    • Term 终止进程
    • Ign 当前进程忽略掉这个信号
    • Core 终止进程,并生成一个Core文件
    • Stop 暂停当前进程
    • Cont 继续执行当前被暂停的进程
  • 信号的几种状态:产生、未决、递达
  • SIGKILL和SIGSTOP信号不能被捕捉、阻塞或者忽略,只能执行默认动作
#include <stdio.h>
#include <string.h>

int main(){

    char * buf;

    strcpy(buf,"hello");

    return 0;
}

这样的一段代码在执行时,会报段错误(核心已转储),因为这里指针是野指针,指向野内存
通过ulimit -a来查看系统信息,可以看到core那行的默认值为0
当我们将其修改为1024,再次执行,虽然还是报段错误,但是会生成一个core文件其中保存了产生错误的信息
通过GDB调试,core-file core语句可以查看core中的内容

正常来说修改完,应该会出现一个core文件,但本人在测试的时候并没有出现,这里注意两个点:

  1. gcc编译出来的文件不能命名为core,重名会无法出现想要的效果
  2. 我的系统是ubuntu,cat /proc/sys/kernel/core_pattern之后输出|/usr/share/apport/apport %p %s %c %d %P %E,查了之后发现ubuntu预装了apport错误收集系统,sudo service apport stop之后就可以了。
  3. 也可以在GDB中查看,直接使用GDB a.out可以看见

(gdb) run
Starting program: /home/fengshou/Linux_server_sufficially/LINUX/day16/a.out

Program received signal SIGSEGV, Segmentation fault.
0x0000555555555135 in main () at core.c:8
8 strcpy(buf, “hello”);
(gdb) Quit

信号相关函数

  • int kill(pid_t pid,int sig):
  • int raise(int sig);
  • void abort(void);
  • unsigned int alarm(unsigned int seconds);
  • int setitimer(int which,const struct itimerval *new_val,struct itimerval * old_value);

kill、raise和abort函数

#include <sys/types.h>
#include <signal.h>

int kill(pid_t pid,int sig);

  • 功能:给某个进程或进程组pid ,发送任何的信号 sig
  • 参数:
    • pid:需要发送给的进程的id
      pid > 0 :发送给指定的进程
      pid = 0 :发送给进程组内所有进程
      pid = -1 :发送给每个有权限接收这个信号的进程
      pid < -1 :发送给每个在进程组中ID是-pid的进程

    • sig:需要发送的信号的编号或者是宏值(推荐使用宏值)
      sig = 0:没有信号发送

eg. kill(getppid(),9);
kill(getpid(),9);

int raise(int sig);

  • 功能:给当前进程发送信号
  • 参数:sig:要发送的信号
  • 返回值:成功返回0,失败返回非0

void abort(void);

  • 功能:发送sIGABRT信号给当前的进程,杀死当前进程
    #include <sys/types.h>
    #include <signal.h>
    #include <stdio.h>
    #include <unistd.h>

    int main(){
        pid_t pid = fork();
        
        if(pid == 0){
            //子进程
            for(int i = 0 ;i < 5;i++){
                printf("child process\n");
                sleep(1);
            }
        }else if(pid > 0){
            //父进程
            printf("parent process\n");
            sleep(2);
            printf("kill child process now\n");
            kill(pid,SIGINT);
        }


        return 0;
    }

在运行的时候,child process会打印2~3次,因为内存中进程是抢时间片运行的,所以在休眠2秒后,肯定会打印2次,概率打印第3次

alarm函数

#include <unistd.h>
unsigned int alarm(unsigned int seconds);

  • 功能:设置定时器(闹钟).函数调用,开始倒计时,当倒计时为0的时候
    函数会给当前的进程发送一个信号:SIGALARM

  • 参数:
    seconds:倒计时的时长,单位:秒。如果参数为0,定时器无效(不进行倒计时,不发信号)
    取消定时器,通过alarm(0).

  • 返回值:
    - 之前没有定时器,返回0
    - 之前有定时器,返回倒计时剩余的时间

  • SIGALARM:默认终止当前的进程,每一个进程都有且只有唯一的一个定时器
    alarm(10); -->返回0
    过了1秒
    alarm(5); --> 返回9 //覆盖了之前的倒计时

alarm(100) -> 该函数是不阻塞的

    #include <unistd.h>
    #include <stdio.h>

int main(){

    int seconds = alarm(5);
    printf("seconds = %d\n",seconds); //0
    
    sleep(2);

    seconds = alarm(2); //不阻塞
    printf("seconds = %d\n",seconds); // 3

    while(1){

    }

    return 0;
}

案例

1秒钟电脑能数几个数

#include <unistd.h>
#include <stdio.h>

int main(){
    alarm(1);
    
    int i = 0;
    while(1){
        printf("%d\n",i++);
    }


    return 0;
}

虽然我们这里设置的是1秒,但是实际运行起来不止1秒才会终止,这是因为程序虽然设置了1秒结束,但是输出的数据会存储在缓冲区,在缓冲区中会稍慢的输出,所以在我们看来程序运行了不止1秒
因为I/O函数效率较低,我们可以尝试将输出到一个文件里,会发现输出的数远超在进程中的

./alarm1 >> a.txt

程序实际运行时间 = 内核时间 + 用户程序 + 消耗的时间
进行文件IO操作的时候比较浪费时间

定时器与进程的状态无关(自然定时法)。无论进程处于什么状态,alarm都会计时

setitimer函数

#include <sys/time.h>
int getitimer(int which, struct itimerval *curr_value);
int setitimer(int which, const struct itimerval *new_value,struct itimerval *old_value);

  • 功能:设置定时器(闹钟)。可以替代alarm函数。进度:微妙us,可以实现周期性定时

  • 参数:

    • which:定时器以什么时间计时
      ITIMER_REAL: 真实时间,时间到达,发送SIGALARM,常用
      ITIMER_VIRTUAL:用户时间,时间到达,发送SIGVTALRM
      ITIMER_PROT:以该进程在用户态和内核态下所消耗的时间来计算,时间到达,发送SIGPROF

    • new_value:设置定时器的属性
      struct itimerval { //定时器的结构体
      struct timeval it_interval; //每个阶段的时间,间隔时间
      struct timeval it_value; //延迟多长时间执行
      };

      struct timeval { //时间的结构体
      time_t tv_sec; //秒数
      suseconds_t tv_usec; //微秒
      };
      过10秒后,每隔2秒定时一次

    • old_value:记录上一次定时的时间参数,一般不使用,指定NULL

  • 返回值:成功返回0,失败返回-1

#include <sys/time.h>
#include <stdio.h>
#include <stdlib.h>
 
 //过3秒以后,每隔2秒定时一次
int main(){

    struct itimerval new_value;

    //设置new_value间隔的值
    new_value.it_interval.tv_sec = 2;
    new_value.it_interval.tv_usec = 0;

    //设置延迟的时间
    new_value.it_value.tv_sec = 3;
    new_value.it_value.tv_usec = 0;

    int ret = setitimer(ITIMER_REAL,&new_value,NULL);//非阻塞的
    printf("定时器开始了。。。");
    if(ret == -1){
        perror("setitimer");
        exit(0);
    }

    getchar();

    return 0;
}

因为这个函数是非阻塞的,所以会输出下面的那句话,然后因为getchar()函数要键盘输入才会继续,所以程序会卡在那边,直到定时结束,从而程序终止

signal信号捕捉函数

  • sighandler_t signal(int signum,sighandler_t handler);
  • int sigaction(int signum,const struct sigaction *act,struct sigaction *oldact);

#include <signal.h>
typedef void (*sighandler_t)(int);
sighandler_t signal(int signum,sighandler_t handler);

  • 功能:设置某个信号的捕捉行为

  • 参数:

    • signum:要捕捉的信号
    • handler:捕捉到信号要如何处理
      • SIG_IGN:忽略信号
      • SIG_DFL:使用信号默认的行为
      • 回调函数:这个函数是内核调用,程序员只负责写,捕捉到信号后如何去处理信号。
        回调函数:
        • 需要程序员实现,提前准备好的,函数的类型根据实际需求,看函数指针的定义
        • 不是程序员调用,而是当信号产生,由内核调用
        • 函数指针是实现回调的手段,函数实现之后,将函数名放到函数指针的位置就可以了
  • 返回值:
    成功,返回上一次注册的信号处理函数的地址。第一次调用返回NULL
    失败,返回SIG_ERR,设置错误号

SIGKILL SIGSTOP不能被捕捉,不能被忽略.

#include <sys/time.h>
#include <stdio.h>
#include <stdlib.h>
#include <signal.h>

void myalarm(int num){
    printf("捕捉到了信号的编号是:%d\n",num);
    printf("XXXXXXXXXX\n");
}

//过3秒以后,每隔2秒定时一次
int main(){
    //注册信号捕捉
    //signal(SIGALRM,SIG_IGN);
    //signal(SIGALRM,SIG_DFL);
    //void (*sighandler_t)(int);函数指针,int类型的参数表示捕捉到的信号的值
    signal(SIGALRM,myalarm);

    struct itimerval new_value;

    //设置间隔的时间
    new_value.it_interval.tv_sec = 2;
    new_value.it_interval.tv_usec = 0;

    //设置延迟的时间,3秒之后开始第一次定时
    new_value.it_value.tv_sec = 3;
    new_value.it_value.tv_usec = 0;

    int retur = setitimer(ITIMER_REAL ,&new_value,NULL);//非阻塞
    printf("定时器开始了...\n");

    if(retur == -1){
        perror ("setitimer");
        exit(0);
    }

    getchar();

    return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/499043.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Spring MVC数据格式化与验证以及国际化和中文乱码处理

目录 Spring MVC数据格式化 基本介绍 ConversionService converters 基本数据类型和字符串自动转换 代码实例 -页面演示方式 创建Monster 类 创建data_valid.jsp 创建MonsterHandler类 创建monster_addUI.jsp 解读: 说明 阶段测试一下 ​编辑 继续完成功能 创建s…

springboot+vue校园疫情防控系统(源码+文档)

风定落花生&#xff0c;歌声逐流水&#xff0c;大家好我是风歌&#xff0c;混迹在java圈的辛苦码农。今天要和大家聊的是一款基于springboot的校园疫情防控系统。项目源码以及部署相关请联系风歌&#xff0c;文末附上联系信息 。 &#x1f495;&#x1f495;作者&#xff1a;风…

理想汽车VS特斯拉,电动汽车正在吞噬世界

来源&#xff1a;猛兽财经 作者&#xff1a;猛兽财经 电动汽车正在吞噬世界 长期以来&#xff0c;电动汽车在美国一直是一项边缘技术&#xff08;当时主要是加州的环保主义者和科技圈在关注&#xff09;。即使现在如日中天的特斯拉(TSLA)&#xff0c;当年也是在成立三年后的20…

【案例1】图书馆管理系统毕业论文

博主介绍&#xff1a; &#x1f680;自媒体 JavaPub 独立维护人&#xff0c;全网粉丝打大于100w&#xff0c;csdn博客专家、java领域优质创作者&#xff0c;51ctoTOP10博主&#xff0c;知乎/掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和副业。&#x1f680; …

npm私有仓库(nexus)-Vue组件发布到nexus

1、创建组件项目 进入项目目录并 执行 npm install 安装依赖&#xff0c;npm run dev 运行&#xff0c;此时浏览器打开 http://localhost:8080 可看到初始化页面 2、在 src 目录下新建测试组件&#xff0c;如下图所示 3、项目目录下新建组件暴露文件 4、修改 package.json pri…

Spring Boot项目使用maven的jib插件打docker使用所需的镜像tar包

jib插件介绍 Jib是一个由Google开发的基于Docker镜像构建的工具&#xff0c;它的Maven插件可以让我们以更高效的方式构建和管理Docker镜像。使用Jib&#xff0c;我们可以摆脱手动编写Dockerfile的繁琐过程&#xff0c;而是直接将我们的Java应用程序打包为镜像&#xff0c;并将…

CPU狂飙900%,该怎么处理?

一位小伙伴面试了 网易&#xff0c;遇到了一个 性能类的面试题&#xff1a;CPU飙升900%&#xff0c;该怎么处理&#xff1f; 可惜的是&#xff0c;以上的问题&#xff0c;这个小伙没有回答理想。 最终&#xff0c;导致他网易之路&#xff0c;终止在二面&#xff0c;非常可惜 …

MySQL学习笔记第九天

第08章聚合函数 上一章讲到了 SQL 单行函数。实际上 SQL 函数还有一类&#xff0c;叫做聚合&#xff08;或聚集、分组&#xff09;函数&#xff0c;它是对一组数据进行汇总的函数&#xff0c;输入的是一组数据的集合&#xff0c;输出的是单个值。 1.聚合函数介绍 什么是聚合…

智能优化算法:袋獾优化算法-附代码

智能优化算法&#xff1a;袋獾优化算法 文章目录 智能优化算法&#xff1a;袋獾优化算法1. 袋獾优化算法1.1 初始化1.2 策略一&#xff1a;以腐肉为食&#xff08;探索阶段&#xff09; 2.实验结果3.参考文献4.Matlab 摘要&#xff1a;袋獾优化算法&#xff08;Tasmanian Devil…

MyBatis官方文档学习笔记(一)

本笔记根据mybatis官方文档顺序学习&#xff0c;根据本笔记可快速掌握mybatis的使用。 1 快速开始 1.1 快速开始 1.1.1 安装 使用MyBatis之前必须要安装mybatis-x.x.x.jar驱动文件到类路径中&#xff0c;如果使用的是Maven工程&#xff0c;则只需要导入下面的依赖即可&…

Java集合-Java集合基础

目录 讲一讲Java集合吧 集合的使用 为什么要使用集合&#xff1f; 如何选用集合? Collection Collection与Collections的区别是什么&#xff1f; Collections.sort和Arrays.sort的实现原理 为何Collection不从Cloneable和Serializable接口继承&#xff1f; 线程安全集…

面向开发人员的 ChatGPT 提示语教程中文版

面向开发人员的 ChatGPT 提示语教程中文版 1. 指南1-1. 提示的指南1-2. 配置1-3. 提示语原则原则 1: 写出清晰而具体的指示技巧 1: 使用分隔符来清楚地表明输入的不同部分技巧 2: 要求提供结构化的输出技巧 3: 要求模型检查条件是否得到满足技巧 4: "少许样本"提示 原…

新魔百盒M304A_增强版2+16G_S905系列_UWE5621DS_卡刷/线刷固件包-当贝桌面

新魔百盒M304A_增强版216G_S905系列_UWE5621DS_卡刷&#xff0f;线刷固件包-当贝桌面 特点&#xff1a; 1、适用于对应型号的电视盒子刷机&#xff1b; 2、开放原厂固件屏蔽的市场安装和u盘安装apk&#xff1b; 3、修改dns&#xff0c;三网通用&#xff1b; 4、大量精简内…

第三十九章 Unity 图像 (Image) UI

本章节&#xff0c;我们介绍一下“Image”UI元素&#xff0c;首先我们创建一个新场景“SampleScene2.unity”。然后我们点击菜单栏“GameObjec”->“UI”->“Image”&#xff0c;如下所示 我们将其放置到了中间位置&#xff0c;检视面板中的参数如下 默认情况下&#xf…

python基础-内置高阶函数map/reduce/filter函数的使用

一、map Python内置函数&#xff0c;用法及说明如下&#xff1a; class map(object):"""map(func, *iterables) --> map objectMake an iterator that computes the function using arguments fromeach of the iterables. Stops when the shortest iterab…

工控机All In One——篇1,pve安装(必读)

工控机All In One——篇1&#xff0c;pve安装 1、背景 满足家庭一机多用&#xff0c;同时满足软路由&#xff08;主旁路由&#xff09;、远程控制、NAS&#xff08;个人存储&#xff09;等功能 2、准备 硬件 1、J4125工控机 2、一个usb分线器 3、鼠标键盘 4、带HDMI或VGA接…

2022 年全国硕士研究生入学统一考试英语(一)试题

2022年全国硕士研究生入学统一考试英语&#xff08;一&#xff09;试题 SectionⅠUse of English Directions: Read the following text. Choose the best word(s) for each numbered blank and mark A, B, C or D on the ANSWER SHEET. (10 points) The idea that plants have…

论文解析-基于 Unity3D 游戏人工智能的研究与应用

1.重写 AgentAction 方法 1.1 重写 AgentAction 方法 这段代码是一个重写了 AgentAction 方法的方法。以下是对每行代码解释&#xff1a; ①public override void AgentAction(float[] vectorAction) 这行代码声明了一个公共的、重写了父类的 AgentAction 方法的方法。它接受…

【力扣-225】用队列实现栈

&#x1f58a;作者 : D. Star. &#x1f4d8;专栏 : 数据结构 &#x1f606;今日小tips : 有些家里可能会有两个同样名字的 WiFi&#xff0c;而其中一个名字的最后有个 5G&#xff0c;那么&#xff0c;这个 5G 和常说的 5G 信号是一回事吗&#xff1f;并不是&#xff0c;无线路…

研报精选230507

目录 【行业230507山西证券】煤炭行业周报&#xff1a;港口煤价趋稳&#xff0c;静待经济信心恢复与迎峰度夏双至 【行业230507山西证券】农业行业周报&#xff1a;建议逢低布局经营稳健和低PB的养殖股 【行业230507头豹研究院】2023年中国无源物联网行业词条报告 【个股230507…