Linux下进程间通信

news2025/1/12 1:07:52

Linux下进程间通信

  • 进程间通信的目的
  • 进程间通信的手段的分类
  • 管道
    • 什么是管道
    • 管道原理
    • 匿名管道
    • 创建匿名管道文件
    • 匿名管道的特点
    • 匿名管道的4种场景
  • 有名管道
    • 有名管道的创建
    • 有名管道总结
    • 命名管道的打开规则
  • system V 共享内存
    • 共享内存原理
    • 建立通信
    • 回收共享内存
    • 开始通信
    • 命令操作共享内存
  • 共享内存与管道的区别

进程间通信的目的

数据传输: 一个进程需要将它自己的数据发送给其它进程;
资源共享: 多个进程间共享同一份资源;
通知事件: 一个进程需要像另一个或者另一组进程发送某个消息,通知它们发生了什么事;(比如:进程终止时要通知父进程);
进程控制: 有些进程希望完全控制另一个进程的执行(如Debug进程),此时控制进程希望能够拦截另一个进程的所有陷入和异常,并能够及时知道它的状态改变;

进程间通信的手段的分类

管道:

  1. 匿名管道;
  2. 有名管道;

System V IPC:

  1. System V 消息队列;
  2. System V 共享内存;
  3. System V 信号量;

POSIX IPC:

  1. POSIX 消息队列;
  2. POSIX 共享内存;
  3. POSIX 信号量;

管道

什么是管道

管道是Unix最古老的进程间通信的手段;
我们把从一个进程连接到另一个进程的一个数据流称为一个管道;

管道原理

根据我们之前学的知识我们可以知道,任何两个进程之间都是相互独立的,我们当前进程无法直接访问另一个进程的数据!
可是,如果我们两个进程都看向同一份资源呢?
比如:A进程看到了一个log.txt文件,进程B也看到了log.txt文件;
那么如果现在A进程是对log.txt文件进行写入操作,而我们的进程B是对log.txt文件进行读取操作,那么是不是对于我B进程来说是不是就拿到了A进程发送出来的数据?
你看我们不就通过这样的一种方式,让两个相互独立的进程就"交流"起来了嘛!
现在我们在对这个log.txt文件稍加修饰:比如:我现在只允许A进程向log.txt文件进行写入操作,不允许进行读取,同时我们规定B文件只能向log.txt文件进行读取操作,不允许进行写,那么你看A进程到B进程的数据流动是不是就有画面感了,数据只允许从A进程流向B进程;
我们把log.txt这种< “只允许单向数据流动的、充当"中间人"的文件”就称作" >管道文件"!
这就是"管道文件"的原理!
在这里插入图片描述
同时通过管道原理的讲解,我们也揭示了进程间通信的本质:就是让不同的进程看到同一份资源!

管道文件分为两种:匿名管道文件、有名管道文件
一般情况下,如果我们只说管道文件的话,一般都是指的匿名管道文件;

匿名管道

人如其名,这个管道文件就是没有名字,同时这个管道文件不是一个磁盘文件,它是一个由OS维护的内存级别的文件,就是说该文件的文件内容和属性并不存储于磁盘上,这些数据就在内存中;
1、既然我任意两个进程之间都能通过匿名管道来进行通信,那么是不是说在内存中可能会同时存在大量的匿名管道文件?
是的!因为在内存中可不止我们这A、B两个进程需要进程进行通信,也有可能C进程与D进程也需要进行通信,他们也有可能采用匿名管道的方式,那么就势必会造成内存中会存在大量的匿名管道文件!
2、那么OS要不要把这些匿名管道文件管理起来呢?
答案:OS作为计算机中的管理者,自然是需要帮助我们把这些匿名管道文件管理起来!为进程提供良好的运行环境;
3、OS如何管理这些匿名管道文件呢?
答案:先描述,再组织! Linux下一切皆文件,匿名管道文件也不例外,OS会用struct file的结构体来描述一个匿名管道文件,这不过这个struct file这个结构体OS是不会为其分配缓冲区的,因为这是个内存级别的文件,不需要将数据刷盘到磁盘,因此匿名管道文件的struct file操作系统不会为其分配缓冲区,除此之外它与那些磁盘文件别无差异!既然都用struct file描述起来了,那么我们只要找到struct file结构体,我们就能找到struct file结构体对应的匿名管道文件!同时OS可能会利用某种数据结构比如(链表、数组等)把这些struct file结构体组织起来,之后OS对于这些匿名管道文件的管理,就变成的对于链表的增删查改!至此OS就完成了对于匿名管道文件的管理!

4、现在问题又来了,既然是匿名管道文件,那么我们如何使用这个这个匿名管道文件呢?
就比如:现在我A进程创建并打开了一个匿名管道文件,我A进程就能拿到匿名文件在我这个进程中的文件描述符,A进程再通过文件描述符来找到匿名管道文件,这没问题!可是我B进程如何打开这个匿名管道文件呢?A进程既然已经创建好了匿名管道文件,那么就自然等待着我们B进程来打开它啊,然后进行欢快的交流,可是这个文件是匿名的、还是内存级别的,我B进程如何利用系统调用open()来打开这个文件啊?我B进程连打开那个匿名文件都不知道,我如何如A进程进行通信?
因此,我们第一步要解决A、B进程如何看到同一份匿名管道文件的问题?
匿名管道文件,没有名字我们不好打开,可是作为创建者A是可以打开的并且准确知道该匿名管道文件的文件描述符的,如果我B进程是A进程的子进程呢?我B进程是不是就会继承父(A)进程的大部分属性,包括文件描述符表:
在这里插入图片描述
这样做到话,A、B两个进程就看到了同一份管道文件资源,就可以进行快乐的通信了!
细心的读者可以发现我们A进程在打开管道文件是分别用的两次不同的方式来打开的:第一次用只写、第二次用只读;
为什么要这样?
我直接利用只读的形式一次打开它不香吗?
我么前面说了,管道文件的数据流动是单向的,如果我们以读写的方式打开匿名管道文件,那么子进程继承下去过后也是以读写的方式打开的匿名管道文件,无法保证数据流动的单向性!因此我们不能用读写的方式来一次性到位!
那为什么分两次就行了呢?
很简单,如果我们想要A进程进行发送数据,B进程进行读取数据,那么在B进程继承A进程过后,马上关闭A进程的读描述符,再马上关闭B进程的写文件描述符,那么这样,A进程就只能对匿名管道文件进程写操作,B进程也就只能对匿名管道文件进行读操作了这样也就保证了数据再管道文件中的单向流动性!(我们也可以让B进行进行写,A进程进行读,只需关闭对应的进程的文件描述就可以了!)

在这里插入图片描述

创建匿名管道文件

话不多说,我们先创建一个匿名管道出来玩一玩:

#include <unistd.h>
int pipe(int pipefd[2]);//调用这个系统调用的进程来创建匿名管道
返回值: 创建成功,返回0;创建失败返回-1;
参数: 输出型参数,就是用于存储管道文件在调用pipe函数的进程中的文件描述符;
pipefd[0]:用于存储管道文件的读端;pipefd[1]:用于存储管道文件的写端;
在这里插入图片描述
根据上面使用匿名管道的原理,父进程有了,父进程也创建了管道文件,那么就开始创建子进程了,让子进程能继承父进程的文件描述符表,然后父子进程就能看见同一份匿名管道资源,但是我们现在让父进程专门执行写,子进程就专门进行读,因此我们需要关闭父进程的pipefd[0]号文件描述符,关闭子进程的pipefd[1]号文件描述符:

#include<iostream>
#include<unistd.h>
#include<cerrno>
#include<cstring>
int main()
{
    int pipefd[2];
        int n=pipe(pipefd);
        if(n!=0)
        {
            //管道文件创建是败
            std::cerr<<"error num:"<<errno<<"error message:"<<strerror(errno)<<std::endl;
            exit(1);
        }
        int id=fork();
        if(id==0)
        {
            //child
            //关闭子进程对于管道文件的写端
            close(pipefd[1]);
            
            //通信……

            close(pipefd[0]);//子进程通信完毕,退出前关闭打开的文件;
            exit(0);
        }
        //father
        close(pipefd[0]);//关闭父进程对于管道文件的读端
       
        //通信……

        close(pipefd[1]);//通信完毕,关闭父进程打开的文件
    return 0;
}

至此,两个独立的进程就可以开始进行通信了;
为了,更加清晰的看到,两个进程之间确实在进行通信,我们可以写一些,简单的测试用语:

#include<iostream>
#include<unistd.h>
#include<cerrno>
#include<cstring>
#include<cstdio>
#include<cstdlib>
int main()
{
    int pipefd[2];
        int n=pipe(pipefd);
        if(n!=0)
        {
            //管道文件创建是败
            std::cerr<<"error num:"<<errno<<"error message:"<<strerror(errno)<<std::endl;
            exit(1);
        }
        int id=fork();
        if(id==0)
        {
            //child
            //关闭子进程对于管道文件的写端
            close(pipefd[1]);
            //通信……
            while (true)
            {
                char buff[1024]={0};
                int n=read(pipefd[0],buff,sizeof(buff)-1);
                printf("我是子进程,我的pid是%d.  父进程=>子进程信息:%s\n",getpid(),buff);
            }
            close(pipefd[0]);//子进程通信完毕,退出前关闭打开的文件;
            exit(0);
        }
        //father
       // close(pipefd[0]);//关闭父进程对于管道文件的读端
        //通信……
        while(true)
        {
            char buff[1024];
            snprintf(buff,sizeof(buff),"你好子进程,我是父进程.我的pid是%d,子进程的pid是%d\n",getpid(),id);
            write(pipefd[1],buff,strlen(buff));//将信息写入管道文件中
            sleep(1);
        }

        close(pipefd[1]);//通信完毕,关闭父进程打开的文件
    return 0;
}

在这里插入图片描述
子进程接收到的信息;

匿名管道的特点

1、管道是半双工的,数据只能向一个方向流动;需要双方通信时,需要建立起两个管道
2、匿名管道文件的本质就是文件,同时匿名管道文件的生命周期随着进程;
3、管道通信,通常用来进行具有“血缘关系”的进程间进行通信,比如:父子进程、兄弟进程、爷孙进程等;
4、在管道通信中,写入的次数和读取的次数,不是严格匹配的,读写次数的多少没有强相关;
5、具有一定的协同能力,让read和write能够按照一定的步骤进行通信,自带同步机制;

匿名管道的4种场景

1、如果我们read完管道里面的数据,对方又不发了,我们(read端)就只能等待:
在这里插入图片描述
在这里插入图片描述
实际上是子进程卡在了read的内部,我们可以用实验证明一下:
在这里插入图片描述实验结果:
在这里插入图片描述
2、如果读端一直不读数据,写端一直写数据,那么当管道文件写满过后,写端则不在写数据,写端会卡在write函数内部;
在这里插入图片描述
实验结果:
在这里插入图片描述
3、当我们关闭了写端,我们又读取完毕了管道内的所有数据,然后read在读,read会读到文件末尾,然后返回0;这也很好理解:我们读端在读的时候发现,写端都被关闭了,那么就再也不会有进程向管道里面写入数据了,就不会再报希望的去等待写端进行写了,而是直接返回0,表示读到了文件末尾;
4、当我们读端关闭了,那么OS将给写端进程发一个13号信号,来终止掉写端进程;

有名管道

管道应用的一个限制就是只能在具有共同祖先(具有亲缘关系)的进程间通信。
如果我们想在不相关的进程之间交换数据,可以使用FIFO文件来做这项工作,它经常被称为命名管道。
命名管道是一种特殊类型的文件

有名管道的创建

命令行上创建一个有名管道文件:
mkfifo filename
在这里插入图片描述
有名管道文件的文件属性是存储在磁盘上的,因此,有名管道文件是具有inode编号的,但是有名管道的文件内容与匿名管道一样是存储在内存中的,这块内存有OS来维护;

当然除了利用命令来创建有名管道文件,我们也可以利用函数在代码中创建:
int mkfifo(const char *filename,mode_t mode);
参数: filename:有名管道的文件名
mode:有名管道文件的的权限,受umask掩码约束
返回值: 成功返回0,失败返回-1;
有名管道只用创建一次就行了;

eg:

//读端进程
#include<iostream>
#include<sys/types.h>
#include<sys/stat.h>
#include<cstring>
#include<fcntl.h>
#include<unistd.h>
int main()
{
    //test1.cc进程创建有名管道
    int n=mkfifo("fifo",0666);//创建有名管道,并且设置有名管道文件的权限是0666,由于受umask码限制,最终fifo文件权限为0664
    if(n==-1)
    {
        //管道文件创建是败
        std::cerr<<"error num:"<<errno<<"error message:"<<strerror(errno)<<std::endl;
        exit(1);
    }
    //打开有名管道文件
    int fd=open("fifo",O_RDONLY);
    //开始通信
    while(true)
    {
        char buff[1024];
        int n=read(fd,buff,sizeof(buff)-1);
        if(n>0)
        buff[n]=0;
        printf("我是test1.cc,test2.cc给我的信息是%s\n",buff);
    }
    close(fd);
    return 0;
}
///
//写端进程
#include<iostream>
#include<sys/types.h>
#include<sys/stat.h>
#include<cstring>
#include<fcntl.h>
#include<unistd.h>
int main()
{
    //test2.cc以写的方式打开有名管道文件
    int fd=open("fifo",O_WRONLY);
    //开始通信
    while(true)
    {
        char buff[1024];
        sprintf(buff,"你好test1.cc,我是test2.cc\n");
        write(fd,buff,strlen(buff));
        sleep(1);
    }
    close(fd);
    return 0;
}

有名管道总结

1、如果我们要创建的有名管道文件已经存在时,我们的mkfifo函数会创建失败:
2、匿名管道的4种场景,同样适用于有名管道
3、有名管道,不仅可以作用与“血缘进程”之间,也可用用于非血缘进程之间;
4、匿名管道,再利用pipe创建的时候就已经帮助我们打开了,不需后续的手动打开;但是对于有名管道来说,我们在利用mkfifo创建有名管道过后,还需要利用open系统调用手动打开;
5、我们在使用完有名管道后,每次都需要手动删除有名管道文件,很是麻烦,我们可以在代码中加入unlink(filename)系统调用,来自动删除有名管道文件,只要程序已结束,有名管道文件就会被自动删除。

命名管道的打开规则

如果当前打开操作是为读而打开FIFO时
O_NONBLOCK disable:阻塞直到有相应进程为写而打开该FIFO
O_NONBLOCK enable:立刻返回成功
如果当前打开操作是为写而打开FIFO时
O_NONBLOCK disable:阻塞直到有相应进程为读而打开该FIFO
O_NONBLOCK enable:立刻返回失败,错误码为ENXIO

system V 共享内存

共享内存区是最快的IPC形式。一旦这样的内存映射到共享它的进程的地址空间,这些进程间数据传递不再涉及到内核,换句话说是进程不再通过执行进入内核的系统调用来传递彼此的数据;

共享内存原理

我们都知道,进程间通信的本质就是让两个互相独立的进程看到同一份资源,那如果两个进程看到的是同一份物理内存呢?那么两个进程之间是不是也能相互进行通信?
我们都知道,每个进程都有属于自己的进程地址空间表和页表,进程所使直接使用的内存地址,实际就是进程地址空间上的虚拟地址,最终还是要通过页表来映射到对应的物理内存上去的;
那么现在如果有两个进程A、B;
A进程映射到物理内存m1,那么B进程是不是也可以映射到物理内存m1呢?
答案是毋庸置疑的!当然可以!
在这里插入图片描述
你看这样我们不就让两个互无关系的两个进程可以进行通信了,因为A、B两个进程都看到了m1这块物理内存,我们把m1这块内存叫做 “共享内存” ;

建立通信

创建共享内存:
int shmget(key_t key, size_t size, int shmflg);
参数:
key: 共享内存的标识符,可由我们用户自己设置,但是要保证其唯一性,不能与之前设置的共享内存的key值冲突;因此我们通常使用 key_t ftok(const char *pathname, int proj_id) 函数来生成key值;
设置这个标识符的原因:就是为了方便后续进程容易看到同一份共享内存;毕竟共享内存不比有名管道,有名字,可以通过名字去查找,共享内存没有名字,后续的进程无法通过名字去查找共享内存,因此我们在利用共享内存进行通信之前必须做出约定,我们准备用那一块共享内存进行通信,而这个key值就是标识共享内存的标识符,相当于共享内存的ID;
size: 需要创建的共享内存的大小,字节为单位;
shmflag: 以何种方式创建共享内促;是位图的一种形式;
常见的选项有:
SHM_CREAT: 创建一个标识符为key的共享内存;如果已经存在一个标识符为key的共享内存,则返回这个共享内存;如果不存在一个标识符为key的共享内存,则创建之,并将其标识符设置为key值,然后返回这个新创建的共享内存;
SHM_CREAT|SHM_EXCL: SHM_EXCL不能单独使用,只能配合着SHM_CREAT来使用;这个组合"键"的意思是:创建一个共享内存,如果已经存在一个标识符为key的共享内存,则直接报错!
如果不存在一个标识符为key的共享内存,则创建之,并将其标识符设置为key值,然后返回这个新创建的共享内存;这也就保证了,如果我们得到了一个共享内存,那么这个共享内存一定是新创建的!
当然,在设置shmflag的时候,我们也可以给共享内存设置权限,因为Linux下一切皆文件!这里设置的权限不受umask掩码的约束;
eg:SHM_CREAT|0666//将共享内存的设置权限为0666
返回值: 如果创建失败,则返回-1;如果成功一个共享内存描述符被返回;
这个共享内存描述符与key是差不多的,都是用来区别不同的共享内存的;如果要说它与key的关系的话,我们可以这样看待:key值就类比于文件系统中的inode编号;shmget返回值就类似于open()函数的返回值(文件描述符);共享内存描述符一样是在上层给用户使用的,而key值是在OS层给OS识别不同共享内存的标识符;

建立映射:
上面的shmget()只是在物理内存中帮我们申请好了一块物理内存罢了,我们这时候还不能直接使用这块物理内存,因为这块物理内存还没有经过进程的页表的映射,映射到进程的地址空间上去:
在这里插入图片描述
因此我们需要shmat()来帮助我们完成,共享内存与进程之间的映射
void *shmat(int shmid, const void *shmaddr, int shmflg);
参数:
shmid: 共享内存描述符,也就是shmget的返回值;告诉shmat需要处理的共享内存是哪个;
shmaddr:用于指定共享内存映射到那个虚拟地址处,一般我们不这么干,我们通常把这个参数设置为nullptr,交给OS自己去决定应该将共享内存映射到那一块虚拟空间;
shmflg: 告诉shmat,它应该以什么样的方式对将共享内存映射到我们的进程地址空间上;也是个位图;
常设置为0:表示将共享内存以读写的方式映射到当前进程上;
SHM_RDONLY:表示以只读的方式映射到当前进程上;
这里设置的映射方式,与我们之前在shmget处为共享内存设置的权限有很大的关系,如果我们刚开始设置的权限中就没有放开写权限,那么我们是不能用shmflg=0的方式来映射到当前进程的,OS会提示我们Permission denied,我们就只能用SHM_RDONLY的方式来建立映射,因为我们设置的共享内存权限中根本没有放开写权限,怎么能谈在以写的方式来链接,这时的共享内存本身就不支持!
返回值: 成功,返回共享内存映射到当前进程地址空间中的虚拟地址;这时的虚拟地址与我们平常代码中的char *之类的地址,并无差异,可以直接使用!失败返回(void *)-1

回收共享内存

取消映射:
既然共享内存创建好了,与进程的映射也建立,那么不同的进程之间,就能愉快的进行通信了;
可是通信完毕过后呢?
我们是不是要回收共享内存?
共享内存可不同管道文件那样,生命周期随进程,进程退出,就销毁;相反,共享内存的生命周期随OS,只要我们不关机,不自己主动去销毁共享内存,那么共享内存就会一直存在!进程退出也会一直存在!
因此在通信结束过后,我们需要主动去销毁共享内存;
共享内存的销毁主要分两步:
1、取消进程与共享内存的映射关系,就是shmat的反操作;
2、当进程与共享内存的链接数为0时,OS就会真正的销毁共享内存!
我们这一步,先讨论取消进程与共享内存的链接:
int shmdt(const void *shmaddr);
参数:

shmaddr:需要取消链接的共享内存在当前进程地址空间中的虚拟地址
返回值:
取消成功,返回0;取消失败,返回-1;

销毁共享内存:
进程与共享内存的链接取消了,接下来就需要销毁了,销毁与创建一样都只需要进行一次即可;
谁创建的共享内存,谁销毁的原则;
int shmctl(int shmid, int cmd, struct shmid_ds *buf);
参数:
shmid:销毁的共享内存
cmd:用于确定shmctl的工作性质;
常见:IPC_RMID:表示shmctl()的工作是销毁共享内存,这里销毁只有当共享内存的链接进程数为0了,才会真正的被销毁,否则,则在描述共享内存的结构体中的shm_perm模式字段的(非标准)SHM_DEST标志将设置IPC_STAT检索的关联数据结构。也就是说共享内存不会被真正的回收,只是会被打上"已死亡"的标签!;
IPC_STAT:那么此时shmctl的工作是获取共享内存的属性信息,这里的属性信息受权限约束!通常将配合第三个参数使用,将获取到的信息放入buf所指的结构体中;
buf:通常配合的IPC_STAT使用,输出型参数,用于存放获取到的共享内存的结构信息;
如果是IPC_RMID模式的话,shmctl的工作是销毁共享内存,不必获取共享内存的状态,buf直接给nullptr就好了;
返回值:成功,返回0;失败返回-1;

开始通信

我们可以写一个简单的demo,来表示两个进程之间通过共享内存来进行通信:

//头文件
#include<iostream>
#include<sys/ipc.h>
#include<sys/shm.h>
#include<sys/types.h>
#include<unistd.h>
#include<cstring>

#ifndef __HAHAHA___
#define __HAHAHA___

#define PATH_NAME "."//确定ftok的第一个参数
#define PROJ_ID 1234 //确定ftok的第二个参数,这两个参数自己随便设置,只要合法就行
#define SHM_SIZE 1024//确定共享内存的大小

#endif
//Master进程进行写
#include"Comm.hpp"
int main()
{
    //创建key值
    key_t key=ftok(PATH_NAME,PROJ_ID);
    //创建共享内存,保证获得的共享内存是新建的
    int shmid=shmget(key,SHM_SIZE,IPC_CREAT|IPC_EXCL|0666);
    //建立映射
    void*begin=shmat(shmid,nullptr,0);
    //开始进行通信
    char *buff=(char*)begin;
    int  cnt=0;
    sleep(2);
    while(cnt<26)
    {
        //直接向共享内存写入数据
        buff[cnt]='A'+cnt;
        buff[++cnt]=0;
        sleep(1);
    }
    //取消链接
    shmdt(begin);
    //销毁共享内存
    shmctl(shmid,IPC_RMID,nullptr);
    return 0;
}
/

//Servant进程进行读
#include"Comm.hpp"
int main()
{
    //创建key值
    key_t key=ftok(PATH_NAME,PROJ_ID);
    //获取共享内存,Master进程已经创建好了,Servant进程只管获取就行了
    int shmid=shmget(key,SHM_SIZE,IPC_CREAT);
    //建立映射
    void*begin=shmat(shmid,nullptr,0);
    //开始进行通信
    char *buff=(char*)begin;
    while(true)
    {
        std::cout<<"I am Servant,MAster give me message is:"<<buff<<std::endl;
        if(strlen(buff)==26)
        break;
        sleep(1);
    }
    //取消链接
    shmdt(begin);
    return 0;
}

//注意运行时机,先运行Master进程,在运行Servant进程
通信效果:
在这里插入图片描述

命令操作共享内存

我们可以利用命令查看我们创建的共享内存:
ipcs -m
查看当前OS下的所有共享内存:
在这里插入图片描述
-q:查看当前OS下的所有消息队列;
-s:查看当前OS下的所有信号量;
不带选项,将一起展示(消息队列、共享内存、信号量);

删除我们创建的共享内存:
ipcrm -m shmid
在这里插入图片描述
ipcrm -s semid:删除信号量;
ipcrm -q mspid:删除消息队列
ipcrm -a 删除所有ipc资源[共享内存|消息队列|信号量]

共享内存与管道的区别

共享内存区是最快的IPC形式:
解释如图:
在这里插入图片描述
通过对比,我们可以知道,很明细使用共享内存来进行通信,要远比使用管道来通信的效率更高!
因为:我们使用共享内存通信,只需要1次copy,而使用管道通信,至少要使用3次copy;copy是很花时间的,很明显使用管道通信的效率更高!

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

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

相关文章

常用的JVM参数选项

目录 打印设置的XX选项及值 堆、栈、方法区等内存大小设置 OutOfMemory相关的选项 垃圾收集器相关选项 GC日志相关选项 其他参数 通过Java代码获取JVM参数 打印设置的XX选项及值 程序运行时JVM默认设置或用户手动设置的XX选项 -XX:PrintCommandLineFlags 打印所有…

Photoshop如何使用绘画和图像修饰之实例演示?

文章目录 0.引言1.给图像添加渐变色效果2.快速创建一副素描画3.清除图像中多余的景物4.快速融合两张图像5.调整图像光影6.人像面部瑕疵修除7.美化眼睛 0.引言 因科研等多场景需要进行绘图处理&#xff0c;笔者对PS进行了学习&#xff0c;本文通过《Photoshop2021入门教程》及其…

LeetCode 与组合数相关的题目

216. 组合总和 III 方法&#xff1a;递归 class Solution { private:vector<vector<int>> res;vector<int> path;void solve(int k, int goal, int cur, int idx) {if (cur > goal) return;if (path.size() k) {if (cur goal) res.push_back(path);re…

小球下落(dropping balls)uva679

题目描述 原文链接 题目链接 上面中文总结一下&#xff1a; D代表这棵树深度&#xff0c;那么一共就有2^d -1 个结点 每个结点从左到右&#xff0c;从上往下&#xff0c;从1开始递增编号&#xff0c;那么也就是说对于结点k来说&#xff0c;左子结点与右子结点的编号分别为…

C/C++每日一练(20230430)

目录 1. 分割回文串 &#x1f31f;&#x1f31f; 2. 六角填数 ※ 3. 查找书籍 &#x1f31f; &#x1f31f; 每日一练刷题专栏 &#x1f31f; Golang每日一练 专栏 Python每日一练 专栏 C/C每日一练 专栏 Java每日一练 专栏 1. 分割回文串 给你一个字符串 s&#x…

unity 渲染性能分析工具

目标 既然要优化&#xff0c;肯定要有个目标&#xff1a; pc上一般要求&#xff1a;一秒渲染60帧 移动端&#xff1a;一秒渲染30帧 这应该是最低的要求&#xff0c;如果游戏运行时&#xff0c;游戏帧率有变化&#xff0c;人眼能够明显的感觉到帧率下降。 优化的首要规则是找到…

CMake | 01 - CMake快速上手(3.26.3)

专栏介绍 本专栏记录了博主入门CMake的笔记。 源码仓库欢迎Star&#xff1a;https://github.com/Mculover666/cmake_study。 一、CMake概述 1. 什么是CMake CMake官网&#xff1a;https://cmake.org/ CMake is an open-source, cross-platform family of tools designed t…

17.计及电转气协同的含碳捕集与垃圾焚烧虚拟电厂优化调度

说明书 MATLAB代码&#xff1a;计及电转气协同的含碳捕集与垃圾焚烧虚拟电厂优化调度 关键词&#xff1a;碳捕集 虚拟电厂 需求响应 优化调度 电转气协同调度 参考文档&#xff1a;《计及电转气协同的含碳捕集与垃圾焚烧虚拟电厂优化调度》完全复现 仿真平台&#xff1a…

【Linux从入门到精通】vim的基本使用各种操作详解

文章目录 一、vim编辑器简单介绍 二、vim编辑器的四种模式 2、1 正常/普通/命令模式(Normal mode) 2、2 插入模式(Insert mode) 2、3 末行模式(last line mode) 三、命令模式的相关操作实例 3、1 光标的相关操作 3、2 文本操作 四、插入模式下的相关操作 五、末行模式下的相关操…

FreeRTOS任务的创建(动态方法和静态方法)

文章目录 前言一、FreeRTOS任务基本概念二、动态创建任务三、静态创建任务四、静态创建任务和动态创建任务的区别五、任务的删除总结 前言 本篇文章将介绍FreeRTOS任务的创建&#xff08;动态方法和静态方法&#xff09;&#xff0c;了解什么是任务和任务的具体创建方法。 一…

文件 IO 操作

文章目录 一 文件1.1 文本模式1.2 二进制模式 二 函数2.1fopen()2.2 getc() 和 putc()2.3 fclose()2.4 fprintf() 和 fscanf()2.5 fgets() 和 fputs()2.6 rewind()2.7 fseek() 和 ftell()2.8 fflush()2.9 fgetpos() 和 fsetpos()2.10 feof() 和 ferror()2.11 ungetc()2.12 setv…

[python][vpython]用vpython实现小球砸弹簧代码

代码&#xff1a; from vpython import * # 加载vpython模块s1 canvas(width1200, height500, backgroundcolor.white, centervector(0, 1, 0)) # 定义画布 L0 1.4 # 定义初始高度 natural_length 0.9 # 设置弹簧原长 base_spring box(posvector(0, 0, 0), sizevector…

solidity 安全 如何阻止重入攻击

什么是可重入攻击&#xff1f; 我们使用合约的过程中&#xff0c;经常会遇到这种情况&#xff0c;智能合约能够调用外部的合约&#xff1b;这些外部合约又可以回调到调用他们的智能合约&#xff1b;在这种情况下&#xff0c;我们说智能合约被重新输入&#xff0c;这种情况被称为…

守护进程Daemon

进程组、对话期和控制终端关系 每个会话有且只有一个前台进程组&#xff0c;但会有0个或者多个后台进程组。产生在控制终端上的输入&#xff08;Input&#xff09;和信号&#xff08;Signal&#xff09;将发送给会话的前台进程组中的所有进程。对于输出&#xff08;Output&…

给大家介绍几个手机冷门但好用的小技巧

技巧一&#xff1a;拍照识别植物 手机的拍照识别植物功能是指在使用手机相机时&#xff0c;可以通过对植物进行拍照&#xff0c;并通过植物识别技术&#xff0c;获取植物的相关信息和资料。其主要优点如下&#xff1a; 方便实用&#xff1a;使用拍照识别植物功能&#xff0c;…

Small Tip: 怎么去Schedule一个Analysis for Office的workbook

workbook的query不能是本地的&#xff0c;也就是说不能是在包$Tmp里面的。这种的没办法在BO里面用SSO。也就没办法Schedule。 前提条件有&#xff1a; 1.BO和BW系统的SSO配置得OK。 如果没有SSO&#xff0c;那么每次打开workbook就会有一个要登录BW的弹窗&#xff0c;这样是…

从零开始学习Linux运维,成为IT领域翘楚(三)

文章目录 &#x1f525;Linux超级用户与伪用户&#x1f525;Linux文件基本属性&#x1f525;Linux权限字与权限操作 &#x1f525;Linux超级用户与伪用户 Linux下用户分为三类&#xff1a;超级用户、普通用户、伪用户 ⭐ 超级用户&#xff1a;用户名为root&#xff0c;具有一切…

初识 OPC

为什么需要OPC&#xff1f; 随着自动化的发展&#xff0c;工厂的自动化程度越来越高&#xff0c;面临着问题&#xff1a; 设备之间&#xff0c;设备与应用程序之间交互数据 不同产商设备/应用程序具有不同接口/通信协议&#xff0c;如何简单快速连接&#xff1f;采集的数据及…

【计算机网络】学习笔记:第四章 网络层【王道考研】更新中....

基于本人观看学习b站王道计算机网络课程所做的笔记&#xff0c;不做任何获利 仅进行交流分享 特此鸣谢王道考研 若有侵权请联系&#xff0c;立删 如果本篇笔记帮助到了你&#xff0c;还请点赞 关注 支持一下 ♡>&#x16966;<)!! 主页专栏有更多&#xff0c;如有疑问欢迎…

计算机组成原理4.2.2汉明码

编码的最小距离 奇校验和偶校验 看1的个数是奇数 还是偶数 汉明码 汉明码的配置 根据不等式&#xff0c;确定增添几位&#xff0c;根据指数放置增添位 汉明码的检错 分不同检测小组 分组规则&#xff1a;哪位为’1‘就是哪组元素。 1号位为‘1’的都是第一组元素&#…