进程间通信/管道/消息队列/共享内存/信号量

news2025/1/17 0:57:45

本文重点目标:

⭐进程间通信介绍⭐
⭐管道⭐
⭐消息队列⭐
⭐共享内存⭐
⭐信号量⭐

1.进程间通信介绍

什么是通信?

通信指的是数据传输、资源共享、通知事件和进程控制。

①数据传输:一个进程需要将它的数据发送给另一个进程
②资源共享:多个进程之间共享同样的资源。
③通知事件:一个进程需要向另一个或一组进程发送消息,通知它(它们)发生了某种事件(如进程终止时要通知父进程)。
④进程控制:有些进程希望完全控制另一个进程的执行(如Debug进程),此时控制进程希望能够拦截另一个进程的所有陷入和异常,并能够及时知道它的状态改变

为什么要通信?

有时候需要多进程协同,让每一个进程专注于自己的事,然后把结果交给另外一个进程去处理。比如使用管道,让多进程协同,简单的有:

cat file | grep "hello"  

我们都知道,管道前面的cat是将file文件的内容现实出来,后面的grep通过管道,拿到file的内容,然后根据内容筛选"hello"指定的内容。也就是说cat负责打印文件内容,grep负责过滤内容,两个进程通过管道 | 连接起来,完成两个进程之间的通信!

因此,通信的目的就是让多进程协同,完成任务。

如何进行进程间通信?

主流的通信的办法有三种:管道、System V进程间通信和POSIX进程间通信。

System V是一种聚焦在本地的一种通信方法,即在一台计算机中进行多进程协同。

POSIX是让通信过程可以跨主机

管道:依托文件系统来处理通信的一套方案。

理解通信的本质

因为进程具有独立性,每个进程自己的资源都只属于自己,因此想要通信,就必须让双方的进程看到同一份资源,而这块资源或空间就是由操作系统直接或间接提供!这也意味着,即使通信的种类很多,但是其本质就是提供资源的模块,是属于操作系统中哪一个模块,比如是由文件系统提供的,那么这个通信的种类就是管道。

总结一下:进程间通信,就是要让不同的进程看到同一份资源,即能够协同使用这些资源,然后进行通信,最后完成任务!

管道

什么是管道?

管道是Unix中最古老的进程间通信的形式。我们把从一个进程连接到另一个进程的一个数据流称为一个“管道”。管道可以分有匿名管道和命名管道,接下来我们先从匿名管道开始学习!

匿名管道 

父子进程是同时指向一个文件,此时这个文件,就是父子进程这两个进程能够同时看到的资源!这就满足了通信的前提。而文件里有文件缓冲区等等,父进程可以向文件缓冲区里面写入内容,子进程又通过文件缓冲区将内容读取出去,这个过程,就完成了一个进程将数据交给另外一个进程,即进程间通信!

其中,使用文件的方式来完成父子进程进行通信,这个文件,称为管道文件!

那么这个管道文件怎么来的呢?

如果是一个普通文件,我们往文件里面写入数据,那么除了把数据放在文件的缓冲区里面,还要把这些数据要刷新到磁盘中。接着进程间通信,一个进程往文件里面写数据,刷新到磁盘,然后操作系统从磁盘里面拿数据,加载到内存,另外一个进程再从内存中拿到进程的上下文,这种方法很慢。因此,对于进程间通信需要用到的文件,压根不需要将数据刷新到磁盘,这就意味着,这个文件是不需要真正的存在于磁盘中,文件对象的各种内容,比如文件的操作方法、内核缓冲区等等,都是操作系统申请的。普通文件被open后,操作系统就会创建这个普通文件的文件对象,然后申请各种东西,现在,即使没有open,操作系统也可以去创建文件对象!因此,我们的管道文件,是内存级别的文件!

此时就可以解释清除了,当一个管道文件的对象被创建出来后,然后把对象的地址填入到一个进程的文件描述符表里面,父进程就能看到这个文件了。接着这个进程fork一下创建子进程,子进程直接拥有跟父进程一模一样的文件描述符表,进而子进程也能跟父进程看到同一个文件。此时,父进程就能通过这个内存级别的文件进行通信!

一般要标定一个文件,是通过文件名来找到指定的文件的,可是当前这个管道文件没有名字,这也的管道文件,叫做匿名文件!

一般而言,管道只能用于单向通信,但父进程需要同时以读写方式打开文件,然后让子进程继承文件描述符表,这也才能灵活地进行通信。当父进程是写方式,那么就关掉读功能,子进程关掉写功能。反过来,父进程是读的方式,那么父进程关掉写功能,子进程关掉读功能。

 总结:通过父进程fork创建子进程,让子进程继承父进程文件描述符,让两个进程看到同一个管道文件,那么这个管道文件是内存级文件,没有名字,这就是匿名管道!

此时,理解了这些,我们就完成了进程间通信的第一步:让不同进程看到同一个资源。那么第二步,自然就是通信啦!接下来,我们通过编写代码来认识通信需要用到的接口。

编写代码/实例代码

#include <unistd.h>
功能:创建一无名管道

int pipe(int fd[2]);

参数
fd:文件描述符数组,其中fd[0]表示读端, fd[1]表示写端
这个参数为输出型参数。在调用pipe的时候,操作系统会打开对于的文件,
得到对应进程的文件描述符表中特点的位置,比如3,4,然后把3和4填充到fd[2]中。

返回值:成功返回0,失败返回错误代码
#include <iostream>
#include <string>
#include <cstring>
#include <cstdio>
#include <unistd.h>
#include <cassert>
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
using namespace std;

int main()
{
    //第一步:创建管道文件,打开读写端
    int fds[2];
    int n = pipe(fds);
    assert(n==0);

    //第二步:创建子进程
    pid_t id = fork();
    assert(id >= 0);
    if(id==0)
    {
        //子进程要写入,即父进程是读,子进程关掉读功能
        close(fds[0]);
        //通信代码
        const char *s = "我是子进程,我正在写入数据";
        int cnt = 0;
       while(true)
       {
            cnt++;
            char buffer[1024];//只有子进程才能看到的缓冲区
            snprintf(buffer,sizeof(buffer),"child->parent say: %s[%d][%d]",s,cnt,getpid());
            write(fds[1],buffer,strlen(buffer));
            sleep(1);//每隔一秒写一次
       }

        close(fds[1]);//可以在最后关掉,也可以不关,因为最后子进程终止了,文件也会跟着关掉
        exit(0);
    }
    //父进程读
    close(fds[1]);
    //父进程的通信代码
    while(true)
    {
        char buffer[1024];
        ssize_t s = read(fds[0],buffer,sizeof(buffer)-1);
        if(s>0)
        {
            buffer[s] = 0;
        }
        cout<<"Get Message# "<<buffer<<" | my pid: "<<getpid()<<endl;
        //注意这里父进程没有sleep
    }


    n = waitpid(id,nullptr,0);
    assert(n==id);
    close(fds[0]);

    //初步猜测,看到是3和4,但是谁3谁4?
    // cout<<"fds[0]: "<<fds[0]<<endl;//3  读端
    // cout<<"fds[1]: "<<fds[1]<<endl;//4  写端

    return 0;
}

结果如下:

 从代码中我们可以看到,子进程是没有输出的代码的,是父进程在输出。从结果中我们可以看到,子进程负责写入,然后将数据交给了父进程,父进程负责读取!这种通信方式,就叫做管道通信方式!

读写特征:

①写入的时候有sleep间隔,读取没sleep间隔

上面的代码有一个细节,那就是我们在子进程的写入的时候,是每隔1秒写入一次,而在父进程的读取数据的时候,没有时间间隔。那么如果我们把写入的间隔改成5秒,结果会是怎么样的?结果会是读取的速度会变慢!

那么,在子进程不进行写入的那5秒之间,父进程在干啥子?父进程在阻塞状态!我们将代码中父进程的代码改写成如下:

    //父进程读
    close(fds[1]);
    //父进程的通信代码
    while(true)
    {
        char buffer[1024];
        cout<<"AAAAAAAAAAA"<<endl;
        ssize_t s = read(fds[0],buffer,sizeof(buffer)-1);
        cout<<"BBBBBBBBBBB"<<endl;
        if(s>0)
        {
            buffer[s] = 0;
        }
        cout<<"Get Message# "<<buffer<<" | my pid: "<<getpid()<<endl;
        //注意这里父进程没有sleep
    }

我们会看到结果:

 第一次的时候,就会将A和B先打印输出,然后将子进程写入的内容进行打印输出,那么在接下来等待的5秒里面,父进程先是打印了第二次的A,然后在read那里进入了阻塞状态!这就证明了,如果管道中没有数据,读端在读的时候,默认阻塞当前在读的进程!

②写入没sleep间隔,读取sleep有间隔

反过来,如果在写入的时候,子进程没有sleep,而在读的时候,父进程每次读完,会sleep一段时间,或者甚至不读,先睡一大觉。那么此时,因为管道是有空间大小的,写满的时候,就不能写啦,再写的话可能会把原来的内容给覆盖了,此时写端会阻塞,等待读端的提取!

如果读端只是sleep一小段时间,而写端不停地写入,此时,因为读的时候,是按buffer的字节个数去读的,也就是说,字节个数有多少,在合法的范围内,读端就会马上读取多少。这就导致了下面这个结果,比如读端休眠2秒,而写端不休眠。

 看到结果显示,它会按行读取,将所有内容全部读取出来!

③子进程写端只写一次数据,并且把自己的写端描述符给关掉

如果子进程把自己的写端关掉,那么就代表着已经读完了。测试代码如下:

    //......
    pid_t id = fork();
    assert(id >= 0);
    if(id==0)
    {
        //子进程要写入,即父进程是读,子进程关掉读功能
        close(fds[0]);
        //通信代码
        const char *s = "我是子进程,我正在写入数据";
        int cnt = 0;
       while(true)
       {
            cnt++;
            char buffer[1024];//只有子进程才能看到的缓冲区
            snprintf(buffer,sizeof(buffer),"child->parent say: %s[%d][%d]",s,cnt,getpid());
            write(fds[1],buffer,strlen(buffer));
            //sleep(10);//每隔一秒写一次
            break;
       }

        close(fds[1]);//可以在最后关掉,也可以不关,因为最后子进程终止了,文件也会跟着关掉
        cout<<"子进程关闭了自己的写端"<<endl;
        exit(0);
    }
    //父进程读
    close(fds[1]);
    //父进程的通信代码
    while(true)
    {
        sleep(2);
        char buffer[1024];
        ssize_t s = read(fds[0],buffer,sizeof(buffer)-1);
        if(s>0)
        {
            buffer[s] = 0;
            cout<<"Get Message# "<<buffer<<" | my pid: "<<getpid()<<endl;
        }
        else if(s==0)
        {
            //读到文件结尾
            cout<<"read: "<<s<<endl;
            break;
        }
        //注意这里父进程没有sleep
    }
    //......

④读端关闭,写

读端已经关闭了,此时写是没有意义了,如果坚持下去,只会浪费操作系统的资源。因此对于这种情况,OS会给写的进程发送信号,去终止写端,子进程也会被杀掉,子进程一旦被杀掉,代表着异常退出,父进程就可以获取到子进程的退出码。下面是测试代码:

代码思路:先让读端读取一次,写端还是不停地写,读端读取一次后,关闭读端,子进程立即被终止,也就是被杀掉了,父进程就能读取到子进程的退出码,获取到子进程退出的信号。

#include <iostream>
#include <string>
#include <cstring>
#include <cstdio>
#include <unistd.h>
#include <cassert>
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
using namespace std;

int main()
{
    //第一步:创建管道文件,打开读写端
    int fds[2];
    int n = pipe(fds);
    assert(n==0);

    //第二步:创建子进程
    pid_t id = fork();
    assert(id >= 0);
    if(id==0)
    {
        //子进程要写入,即父进程是读,子进程关掉读功能
        close(fds[0]);
        //通信代码
        const char *s = "我是子进程,我正在写入数据";
        int cnt = 0;
       while(true)
       {
            cnt++;
            char buffer[1024];//只有子进程才能看到的缓冲区
            snprintf(buffer,sizeof(buffer),"child->parent say: %s[%d][%d]",s,cnt,getpid());
            write(fds[1],buffer,strlen(buffer));
            //sleep(10);//每隔一秒写一次
            //break;
       }

        close(fds[1]);//可以在最后关掉,也可以不关,因为最后子进程终止了,文件也会跟着关掉
        cout<<"子进程关闭了自己的写端"<<endl;
        exit(0);
    }
    //父进程读
    close(fds[1]);
    //父进程的通信代码
    while(true)
    {
        sleep(2);
        char buffer[1024];
        ssize_t s = read(fds[0],buffer,sizeof(buffer)-1);
        if(s>0)
        {
            buffer[s] = 0;
            cout<<"Get Message# "<<buffer<<" | my pid: "<<getpid()<<endl;
        }
        else if(s==0)
        {
            //读到文件结尾
            cout<<"read: "<<s<<endl;
            break;
        }
        break;
    }
    close(fds[0]);//父进程读端只读一次后,关闭读端
    cout<<"父进程关闭读端"<<endl;
    int status = 0;
    n = waitpid(id,&status,0);
    assert(n==id);
    cout<<"pid->"<<n<<" : "<<(status & 0x7F) <<endl;

    return 0;
}

从结果我们可以看到,先是等待两秒,再进行第一次的读取,这个读取是按行读取的。第二次的时候,读端被关闭了,子进程终止,并且发送的是13号信号。

 管道特征

 ①只能用于具有共同祖先的进程(具有亲缘关系的进程)之间进行通信;通常,一个管道由一个进程创建,然后该进程调用fork,此后父、子进程之间就可应用该管道。


②管道提供流式服务


③一般而言,进程退出,管道释放,所以管道的生命周期随进程


④一般而言,内核会对管道操作进行同步与互斥,是对共享资源机制的一种包含方案。两个独立进程会很照顾对方的感受,你读累了,那我写满后就不写了;你写累了,不写了,那我也不催促你,我也不读,等你不累了写了再读。这也就可以避免资源出现错误。


⑤管道是半双工(单向通信的特殊概念)的,数据只能向一个方向流动;需要双方通信时,需要建立起两个管道

命名管道

什么是命名管道?

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

创建一个命名管道

从命令行上创建:

$ mkfifo filename

从程序里面创建:

#include<sys/types.h>

#include<sys/stat.h>

int mkfifo(const char *filename,mode_t mode);

参数:第一个参数(const char *filename)填写的是我们要创建的管道文件的路径
第二个参数(mode_t mode)填写的是这个文件的权限,一般为0666.

返回值:创建成功返回0


比如下面的代码:

 
int main()
{
        mkfifo("p2", 0644);
        return 0;
}

 命名管道自然是一个独立文件,以p开头,以p开头的文件,称为管道,没错,这个文件,跟书华竖划线 | 一样,是管道!它可以让不是亲属关系的进程进行通信,让其看到同一份资源。那么,命名管道是如何做到的呢?

因为文件名+路径 = 唯一性。因此通过让不同进程打开指定名称(路径+文件名)的同一个文件,这样就能让不同进程找到同一份文件,能够看到同一份资源了,对比匿名管道,匿名管道是通过子进程继承文件描述符的方式确定管道文件的唯一性。

匿名管道:通过子进程继承父继承的文件描述符表来确定管道文件的唯一性。

命名管道:通过打开指定名称(路径+文件名 = 唯一性)的同一个文件。

编写代码

先创建2个cpp文件,表示两个没有亲属关系的进程,然后创建1个头文件,用于管道文件的创建。

两个cpp文件:server.cpp用于读取数据,client.cpp用于写入数据。

读取数读端的server.cpp代码:

#include "comm.hpp"

using namespace std;

int main()
{
    //创建文件,并且判断是否创建成功
    bool r = createFifo(NAMED_PIPE);
    assert(r);
    (void)r;

    int rfd = open(NAMED_PIPE,O_RDONLY);
    if(rfd < 0)exit(1);//创建失败

    //read
    char buffer[1024];
    while(true)
    {
        ssize_t s =read(rfd,buffer,sizeof(buffer));
        if(s>0)//读取成功,打印
        {
            buffer[s] = 0;
            cout<<buffer<<endl;
        }
        else//读取失败,退出
        {
            std::cout << "client quit, me too!" << std::endl;
            break;
        }
    }

    close(rfd);//关闭文件

    removeFifo(NAMED_PIPE);//最后删除文件

    return 0;
}

写端client.cpp代码:

#include "comm.hpp"

int main()
{
    int wfd = open(NAMED_PIPE,O_WRONLY);
    if(wfd < 0) exit(1);
    char buffer[1024];

    while(true)
    {
        fgets(buffer,sizeof(buffer),stdin);
        if(strlen(buffer)>0)
        {
            buffer[strlen(buffer)-1] = 0;//去掉换行
        }
        ssize_t w = write(wfd,buffer,sizeof(buffer));
        assert(w==strlen(buffer));
        (void)w;
    }

    close(wfd);
    return 0;
}

头文件:

#include <iostream>
#include <string>
#include <cstring>
#include <cerrno>
#include <cassert>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

#define NAMED_PIPE "/wjmhlh/name_pipe"  //管道文件的路径,建立在当前路径

//创建管道文件
bool createFifo(const string &path)
{
    umask(0);
    //创建管道文件
    int n = mkfifo(path.c_str(),0600);
    if(n==0)//管道文件创建成功
    {
        return true;
    }
    else
    {
        std::cout << "errno: " << errno << " err string: " << strerror(errno) << std::endl;
        return false;
    }
}

//删除文件
void removeFifo(const string &path)
{
    int n = unlink(path.c_str());
    assert(n==0);
    (void)n;
}

未完待续......

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

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

相关文章

【GIS】高分辨率遥感影像智能解译

1 绪论 随着航空科技工业的不断成熟与发展&#xff0c;我国遥感卫星研制能力不断攀升&#xff0c;发射数量逐年提高&#xff0c;在轨运行的遥感卫星为社会生产及居民日常生活提供了巨大的支持与便利。我国目前同时在轨运行的遥感卫星数量已超过60颗&#xff0c;每天获取并传回…

LeetCode[990]等式方程式的可满足性

难度&#xff1a;中等题目&#xff1a;给定一个由表示变量之间关系的字符串方程组成的数组&#xff0c;每个字符串方程 equations[i]的长度为 4&#xff0c;并采用两种不同的形式之一&#xff1a;"ab"或 "a!b"。在这里&#xff0c;a 和 b 是小写字母&#…

【Python学习003】高效数据结构-列表

【Python学习003】高效数据结构-列表 我们的公众号是【朝阳三只大明白】&#xff0c;满满全是干货&#xff0c;分享近期的学习知识以及个人总结&#xff08;包括读研和IT&#xff09;&#xff0c;希望大家一起努力&#xff0c;一起加油&#xff01;求关注&#xff01;&#xf…

浏览器使用本地硬盘上的JS文档

网页通过<script>标记可以引入在线的js文档&#xff0c;只有把网页保存在本地硬盘才能使用相对路径引入本地js文档&#xff0c;普通浏览器难以实现在线页面上引入本地js文档。怎么解决这个问题呢&#xff1f;首先准备好需要引入的JS文档&#xff0c;可以是成熟的JS库&…

【docker概念和实践 4】(3)本地镜像提交到阿里云

一、说明 本篇讲述如何将自己制作的容器转化成镜像&#xff0c;再将镜像推送到远端阿里云上。 二、制造一个本地容器 2.1 从远端仓库拉取ubuntu镜像 使用镜像拉取语句&#xff1a; docker pull ubuntu:20.04 docker run -it ubuntu&#xff1a;20.04 bash 通过以上两句&…

LeetCode[200]岛屿数量

难度&#xff1a;中等题目&#xff1a;给你一个由 1&#xff08;陆地&#xff09;和 0&#xff08;水&#xff09;组成的的二维网格&#xff0c;请你计算网格中岛屿的数量。岛屿总是被水包围&#xff0c;并且每座岛屿只能由水平方向和/或竖直方向上相邻的陆地连接形成。此外&am…

5-4中央处理器-控制器的功能和工作原理

文章目录一.控制器的结构和功能二.硬布线控制器1.硬布线控制单元图2.微操作命令分析3.CPU的控制方式&#xff08;1&#xff09;同步控制方式&#xff08;2&#xff09;异步控制方式&#xff08;3&#xff09;联合控制方式4.硬布线控制单元的设计步骤&#xff08;1&#xff09;分…

qsort函数用法 + 模拟实现qsort函数

&#x1f466;个人主页&#xff1a;Weraphael ✍&#x1f3fb;作者简介&#xff1a;目前是C语言学习者 ✈️专栏&#xff1a;【C/C】算法 &#x1f40b; 希望大家多多支持&#xff0c;咱一起进步&#xff01;&#x1f601; 如果文章对你有帮助的话 欢迎 评论&#x1f4ac; 点赞…

论文笔记目录(ver2.0)

1 时间序列 1.1 时间序列预测 论文名称来源主要内容论文笔记&#xff1a;DCRNN &#xff08;Diffusion Convolutional Recurrent Neural Network: Data-Driven Traffic Forecasting&#xff09;_UQI-LIUWJ的博客-CSDN博客iclr 2017使用双向扩散卷积GRU&#xff0c;建模空间和…

目标跟踪心得篇五:MOT数据集标注、TrackEval的使用、DarkLabel不能自动跟踪解决方案及如何在MMTracking上输出跟踪评测

跟踪方向的标注成本非常很大的 ,那么我们如何尽可能一次性弄好呢? 所选标注工具:DarkLabel DarkLabel是一个轻量的视频标注软件,尤其做MOT任务非常友好,其标注可以通过脚本转化为标准的目标检测数据集格式、ReID数据集格式和MOT数据集格式。 使用之前: darklabel.yml:保…

传参的理解

前言 当我们调用函数的时候&#xff0c;参数是怎么传递给被调用方的&#xff0c;有想过这个问题吗&#xff1f;传递不同大小的参数对调用方式有影响吗&#xff1f;本文将带你探究这些问题&#xff0c;阅读本文需要对函数栈帧有一定的理解&#xff0c;并了解基本的汇编指令。 …

傅一平:2022年我的私人书单

2022年过去了&#xff0c;推荐我的TOP 10 书单&#xff0c;同时附上我的一句话评语和豆瓣的评分&#xff0c;这些书代表了我学习的方向&#xff0c;包括学习方法、思考方法、数据治理、数字化转型、系统架构、职场管理、个人修养、生活态度等。TOP 1 学习究竟是什么一句话评语…

【Ajax】HTTP超文本传输协议

一、HTTP协议简介什么是通信通信&#xff0c;就是信息的传递和交换。通信三要素&#xff1a;通信的主体通信的内容通信的方式1.1 现实生活中的通信案例&#xff1a;张三要把自己考清北大学的好消息写信告诉自己的好朋友李四。其中&#xff1a;通信的主体是张三和李四&#xff1…

Linux网络:应用层之HTTP协议

文章目录一、应用层1.协议2.网络版计算器二、HTTP 协议1. URL2. HTTP 协议格式3.查看 HTTP 请求4.发送 HTTP 响应5. HTTP 的方法6. HTTP 的状态码7. HTTP 的版本8. HTTP 常见 Header9. Cookie 与 session三、HTTP 与 HTTPS一、应用层 我们程序员写的一个个解决实际问题&#x…

jvm 堆 栈中存什么?

数据类型 Java虚拟机中&#xff0c;数据类型可以分为两类&#xff1a;基本类型和引用类型。基本类型的变量保存原始值&#xff0c;即&#xff1a;他代表的 值就是数值本身&#xff1b;而引用类型的变量保存引用值。“引用值”代表了某个对象的引用&#xff0c;而不是对象本身&…

矩阵理论复习(六)

Q代表有理数&#xff0c;即整数和小数部分有限的分数和小数部分无限循环的分数。无限不循环的小数就是无理数。所有无理数和有理数加起来就是实数集R。与实数对应的就是虚数。 数域的定义 线性空间的定义 线性空间的基和维数 子空间的定义 子空间的判别方法 最常见的…

【唐诗学习】二、初唐诗词领路人

二、初唐诗词领路人 唐朝之前的主流诗人都是在宫廷混口饭吃&#xff0c;他们整天围着皇帝转&#xff0c;写的大多是宫廷奢靡的生活&#xff0c;还会拍皇帝马屁。主流诗人受前朝影响很大&#xff0c;就这么发展到了初唐。照这个剧情发展下去&#xff0c;诗歌迟早要完蛋。 可有些…

狂神聊Git~

版本控制&#xff1a; 版本控制的概念: 它是一种在开发的过程中用于管理我们对文件&#xff0c;目录或工程等内容的修改历史&#xff0c;方便我们查看历史记录&#xff0c;备份以便恢复以前的版本的软件工程技术 版本控制的作用: 用于管理多人协同开发项目的技术 实现跨区…

Tomcat进程占用CPU过高怎么办?

在性能优化这个主题里&#xff0c;前面我们聊过了Tomcat的内存问题和网络相关的问题&#xff0c;接下来我们看一下CPU的问题&#xff0c;CPU资源经常会成为系统性能的一个瓶颈&#xff0c;这其中的原因是多方面的&#xff0c;可能是内存泄漏导致频繁GC&#xff0c;进而引起CPU使…

Linux命令--查看发行版本/内核版本的方法

原文网址&#xff1a;Linux命令--查看发行版本/内核版本的方法_IT利刃出鞘的博客-CSDN博客 简介 本文介绍Linux查看发行版本和内核版本的方法。 查看发行版本 cat /etc/lsb_release 说明 这个命令适用于大部分linux发行版本&#xff08;除了redhat和centos等&#xff09; …