Linux进程间通信 pipe 实现线程池 命名管道 实现打印日志 共享内存代码验证 消息队列 信号量

文章目录

    • 前言
    • 管道
      • 匿名管道
    • pipe
    • 测试管道接口 --> 代码验证
      • 管道的4种情况
      • 管道的5种特征
    • 线程池案例
      • 代码实现:
        • ProcessPool.cc
        • Task.hpp
        • 检测脚本
        • makefile
    • 命名管道
      • 代码演示:
        • makefile
        • namedPipe.hpp
        • server.cc
        • client.cc
      • 实现日志
        • Log.hpp
    • 共享内存
      • 共享内存原理
      • 补充指令集(IPC的指令)
        • shmget
        • 谈谈key
        • ftok
        • ipcs
        • shmat
        • shmdt
        • shmctl
      • 代码验证(使用共享内存的相关接口)
        • Shm.hpp
        • client.cc
        • server.cc
    • system V消息队列
    • system V信号量
    • 进程互斥

前言

  1. 进程为什么要通信?
  • 进程也是需要某种协同的,所以如何协同的前提条件就是通信
  • 数据是由类别的,通知就绪的,有一些就是单纯的要传递给我数据,控制相关的信息

事实:进程是具有独立性的。进程 = 内核数据结构 + 代码和数据

  1. 进程如何通信?
  • 进程间通信的本质,必须让不同的进程看到同一份“资源
  • 资源”就是特定形式的内存空间
  • 这个资源是由操作系统来提供的,那么为什么不是我们两个进程中的一个?假设一个进程提供,这个资源属于谁,这个进程独有,破坏进程独立性,第三方空间。
  • 我们进程访问这个空间,进行通信,本质就是访问操作系统!
    • 进程代表的就是用户,“资源”从创建,使用(一般),释放资源我们需要使用系统调用接口
    • 从底层设计,从接口设计,都要由操作系统独立设计,一般操作系统会有一个独立的通信模块,属于文件系统 (IPC通信模块),标准(system V && posix)
    • system V:三种方式:消息队列、共享内存、信号量

还有一种就是基于文件级别的通信方式---->管道

管道

  • 管道是Unix中最古老的进程间通信的形式。
  • 我们把从一个进程连接到另一个进程的一个数据流称为一个“管道”

匿名管道

  • 一个文件被打开两次struct_file是要被创建两个的

  • 第二次打开同一个文件的时候,不需要再次加载文件

  • 在创建一个子进程的时候,不会再次加载文件,因为进程要保持独立性,和文件没有关系

为什么父子进程会向同一个显示器终端打印数据

  • 子进程会继承父进程的文件描述符表,会指向同一个文件
  • 进程会默认打开三个标准输入输出错误:0,1,2,怎么做到的?
    • 其实我们所有的在命令行中都是bash的子进程,bash打开了,所有的子进程默认也就打开了,我们只要做好约定即可

为什么我们主动close(0/1/2),不影响父进程继续使用显示器文件呢?

  • 其实在struct_file里面包含了一个引用计数,是一个内核级的,这也就能解释了

进程间通信的本质,必须让不同的进程看到同一份“资源”,这个资源是由操作系统来分配的,我们看到的同一份“资源”就是内核级的文件缓冲区

在这里插入图片描述

  • 管道只允许单向通信,因为它简单

  • 那么如何通信呢?

    • 子进程想写就关闭读的文件描述符(3),父进程就关闭写的文件描述符(4),此时,父进程就可以通过3号描述符进行,子进程就可以通过4号文件描述符进行,双方就可以写入同一个管道文件了。

父子既然要关闭不需要的fd,为什么曾经要打开呢?可以不关吗?

  • 如果只打开一个文件描述符的话,未来子进程继承的时候也就会继承一个,那么以读方式打开,继承只能继承读,一个管道不能同时存在读写,我们也不能以读写的方式打开,因为管道是单向通信的,万一失误了呢?这个方式很不好~~
  • 所以总的来说就是为了让子进程继承下去
  • 可以不关吗?可以~~,建议关了,万一读误写了呢?

还有就是为什么我们两个进程通信的时候,只是在内核级文件缓冲区,而不需要刷新到磁盘,所有虽然管道可以复用,但是还是要重新设计一下

在这里插入图片描述

pipe

  • 接下来我们可以使用pipe来打开管道
int pipe(int pipefd[2]);
  • pipefd是一个输出形参数
  • 不需要文件路径和文件名(匿名文件/匿名管道

在这里插入图片描述

  • 成功返回0,失败返回-1,错误码被设置

在这里插入图片描述

如果我想要双向通信呢?

  • 两个管道

为什么单向通信?

  • 因为简单,只让它进行单向通信,符合这样的特点所以就叫管道

测试管道接口 --> 代码验证

#include <iostream>
#include <unistd.h>
#include <cerrno>
#include <cstring>
#include <string>
#include <sys/wait.h>
#include <sys/types.h>const int size = 1024;std::string getOtherMessage()
{// 计数static int cnt = 0;std::string massageid = std::to_string(cnt);cnt++;// 获取pidpid_t self_id = getpid();std::string id = std::to_string(self_id);std::string massage = "massage:";massage += massageid;massage += " my pid is: ";massage += id;return massage;
}void SubProcessWrite(int wfd)
{int pipesize = 0;std::string massage = "father, I am your son prcess!";char c = 'A';while (true){std::cerr << "++++++++++++++++++++++++++++++++++++++++++" << std::endl;std::string info = massage + getOtherMessage(); // 子进程写给父进程的消息write(wfd, info.c_str(), info.size());std::cerr << info << std::endl;// write(wfd,&c,1);// std::cout << "pipesize: " << ++pipesize << std::endl;// c++;// if(c == 'G') break;sleep(1);}std::cout << "child quit..." << std::endl;
}void FatherProcessRead(int rfd)
{char inbuffer[size];while (true){sleep(2);std::cout << "-------------------------------------------" << std::endl;ssize_t n = read(rfd, inbuffer, sizeof(inbuffer) - 1); // 注意是sizeofif (n > 0){inbuffer[n] = 0;std::cout << "father get massage: " << inbuffer << std::endl;}else if (n == 0){// read返回值为0表示写端关闭了,读到了文件的结尾std::cout << "client quit, father get return val: " << n << "father quit tool!" << std::endl;break;}else if (n < 0){std::cerr << " read error" << std::endl;break;}}
}int main()
{// 1.创建管道int pipefd[2]; // pipefd里的0号下标保存的是读,1号下标保存的是写int n = pipe(pipefd);if (n != 0){std::cerr << "errno:" << errno << ":"<< "errstring:" << strerror(errno) << std::endl;return 1;}std::cout << "pipefd[0]: " << pipefd[0] << " pipefd[1]: " << pipefd[1] << std::endl;// 2.fork创建出父子进程pid_t id = fork();if (id == 0){std::cout << "子进程关闭不需要的fd了,准备发消息了" << std::endl;sleep(1);// 子进程 --- write// 3.关闭不需要的文件秒速符close(pipefd[0]);SubProcessWrite(pipefd[1]);// 不用了就关闭close(pipefd[1]);exit(0);}std::cout << "父进程关闭不需要的fd了,准备收消息了" << std::endl;// 父进程 --- readsleep(1);// 3.关闭不需要的文件秒速符close(pipefd[1]);FatherProcessRead(pipefd[0]);// 不用了就关闭close(pipefd[0]);int status = 0;pid_t rid = waitpid(id, nullptr, 0);if (rid > 0){std::cout << "wait child process done, exit sig: " << (status & 0x7f) << std::endl;std::cout << "wait child process done, exit code(ign): " << ((status >> 8) & 0xFF) << std::endl;}return 0;
}

管道的4种情况

  1. 如果管道内部是空的 && write fd没有关闭,读取条件不具备,读进程会被阻塞(wait–>读取条件具备<–写入数据)
  2. 管道被写满 && read fd不读取且没有关闭,管道被写满,写进程会被阻塞(管道被写满—>条件不具备) —> wait 写条件具备就读数据
  3. 管道一直在读 && 写端关闭了wfd,读端read返回值会读到0,表示读到了文件末尾
  4. rfd直接关闭写端wfd一直在进行写入,写端进程会被操作系统直接使用13信号关掉,相当于进程出现了异常

管道的5种特征

  1. 匿名管道,只是用来进行具有血缘关系的进程之间进行通信,具有明显的顺序性
  2. 管道内部,自带进程之间的同步机制(同步机制就是多执行流执行代码的时候,具有明显顺序性)
  3. 管道文件的生命周期是随进程的
  4. 管道文件在通信的时候,是面向字节流的,write的次数和读取的次数不是一一匹配的
  5. 管道的通信模式,是一种特殊的半双工模式

可以通过上面的代码进行测试在ubuntu20.04管道的大小是4kb
我们平时在命令行中使用的|就是匿名管道


线程池案例

在这里插入图片描述

  • 管道里没有数据,worker进程就在阻塞等待,等待任务的到来

  • master向哪一个管道进行写入,就是唤醒哪一个子进程来处理任务

  • 父进程要进行后端任务划分的负载均衡

代码实现:

ProcessPool.cc
#include "Task.hpp"class Channel
{
public:Channel(int wfd, pid_t id, const std::string &name): _wfd(wfd), _subprocessid(id), _name(name) {}int GetWfd() { return _wfd; }pid_t GetProcessId() { return _subprocessid; }std::string GetName() { return _name; }// 等待子进程void Wait(){pid_t rid = waitpid(_subprocessid, nullptr, 0);if (rid > 0){std::cout << "wait " << rid << " success" << std::endl;}}// 关闭void CloseChannel(){close(_wfd);}~Channel() {}private:int _wfd;pid_t _subprocessid;std::string _name;
};void CreateChannelAndSub(int num, std::vector<Channel> *channels, task_t task)
{for (int i = 0; i < num; i++){// 1. 创建管道int pipefd[2] = {0};int n = pipe(pipefd);if (n < 0)exit(-1);// 2. 创建子进程pid_t id = fork();if (id == 0){// 关闭多余的写端if (!channels->empty()){for (auto &channel : *channels){// 关闭等待channel.CloseChannel();channel.Wait();}}// child --- readclose(pipefd[1]);dup2(pipefd[0], 0); // 将管道的读端,重定向到标准输入task();close(pipefd[0]);exit(0);}// parent// 3.构建一个channel名称std::string channel_name = "Channel-" + std::to_string(i);// a. 子进程的pid b. 父进程关心的管道的w端channels->push_back(Channel(pipefd[1], id, channel_name));close(pipefd[0]); // 父进程关心的管道的w端}
}int NextChannel(int channels)
{static int next = 0;int channel = next;next++;next %= channels;return channel;
}void SendTaskCommand(Channel &channel, int taskcommand)
{write(channel.GetWfd(), &taskcommand, sizeof(taskcommand));
}void ctrlProcessOnce(std::vector<Channel> &channels)
{sleep(1);// a. 选择一个任务int taskCommand = SelectTask();// b. 选择一个信道和进程int channel_index = NextChannel(channels.size());// c. 发送任务SendTaskCommand(channels[channel_index], taskCommand);std::cout << std::endl;std::cout << "taskcommand: " << taskCommand << " channel: "<< channels[channel_index].GetName() << " sub process: " << channels[channel_index].GetProcessId() << std::endl;
}void ctrlProcess(std::vector<Channel> &channels, int times = -1)
{if (times > 0){while (times--){ctrlProcessOnce(channels);}}else{while (true){ctrlProcessOnce(channels);}}
}void CleanUpChannel(std::vector<Channel> &channels)
{// 方法一:// for (auto &ch : channels)// {//     ch.CloseChannel();// }// for (auto &ch : channels)// {//     ch.Wait();// }// 方法二:// int last = channels.size() - 1;// for (int i = last; i >= 0; i--)// {//     close(channels[i].GetWfd());//     waitpid(channels[i].GetProcessId(), nullptr, 0);// }// 方法三:for (auto &ch : channels){ch.CloseChannel();ch.Wait();}
}int main(int argc, char *argv[])
{if (argc != 2){std::cerr << "Usage: " << argv[0] << " processnum" << std::endl;return 1;}int num = std::stoi(argv[1]);LoadTask();std::vector<Channel> channels;// 1. 创建信道和子进程CreateChannelAndSub(num, &channels, work);// 2. 通过channel控制子进程ctrlProcess(channels, num);// 3. 回收管道和子进程. a. 关闭所有的写端 b. 回收子进程CleanUpChannel(channels);// sleep(100);return 0;
}
Task.hpp
#pragma once#include <iostream>
#include <ctime>
#include <cstdlib>
#include <unistd.h>
#include <string>
#include <vector>
#include <sys/types.h>
#include <sys/wait.h>#define TaskNum 3typedef void (*task_t)(); // 函数指针
task_t tasks[TaskNum];    // 4个任务void Print()
{std::cout << "I am print task" << std::endl;
}void DownLoad()
{std::cout << "I am DownLoad task" << std::endl;
}void Flush()
{std::cout << "I am Flush task" << std::endl;
}void LoadTask()
{srand(time(nullptr) ^ getpid() ^ 17777);tasks[0] = Print;tasks[1] = DownLoad;tasks[2] = Flush;
}int SelectTask()
{return rand() % TaskNum;
}void ExcuteTask(int number)
{if (number < 0 || number > 2)return;tasks[number]();
}void work()
{while (true){int command = 0;int n = read(0, &command, sizeof(command));if (n == sizeof(int)){std::cout << "pid is : " << getpid() << " handler task" << std::endl;ExcuteTask(command);}else if (n == 0){std::cout << "sub process : " << getpid() << " quit" << std::endl;break;}}
}void work1()
{while (true){int command = 0;int n = read(0, &command, sizeof(command));if (n == sizeof(int)){std::cout << "pid is : " << getpid() << " handler task" << std::endl;ExcuteTask(command);}else if (n == 0){std::cout << "sub process : " << getpid() << " quit" << std::endl;break;}}
}void work2()
{while (true){int command = 0;int n = read(0, &command, sizeof(command));if (n == sizeof(int)){std::cout << "pid is : " << getpid() << " handler task" << std::endl;ExcuteTask(command);}else if (n == 0){std::cout << "sub process : " << getpid() << " quit" << std::endl;break;}}
}
检测脚本
while :; do ps ajx | head -1 &&  ps ajx | grep -i Processpool | grep -v grep; echo "------------------------"; sleep 1;done
makefile
processpool: ProcessPool.ccg++ -o $@ $^ -std=c++11
.PHONY:clean
clean:rm -f processpool

在这里插入图片描述

命名管道

  • 命名管道可以从命令行上创建,命令行方法是使用下面这个命令:
mkfifo filename

在这里插入图片描述

  • 文件名+路径就可以看到同一份资源

代码演示:

makefile
.PHONY:all
all:server clientserver:server.ccg++ -o $@ $^ -g -std=c++11
client:client.ccg++ -o $@ $^ -g -std=c++11.PHONY:clean
clean:rm -rf server client
namedPipe.hpp
#include <iostream>
#include <cstdio>
#include <cerrno>
#include <string>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>const std::string comm_path = "./myfifo";
#define DefaultFd -1
#define Creater 1
#define User 2
#define Read O_RDONLY
#define Write O_WRONLY
#define BaseSize 4096class NamePiped
{
private:bool OpenNamedPipe(int mode){_fd = open(_fifo_path.c_str(), mode);if (_fd < 0)return false;return true;}public:NamePiped(const std::string &path, int who): _fifo_path(path), _id(who), _fd(DefaultFd){if (_id == Creater){int res = mkfifo(_fifo_path.c_str(), 0666);if (res != 0)perror("mkfifo");std::cout << "creater create named pipe" << std::endl;}}bool OpenForRead(){return OpenNamedPipe(Read);}bool OpenForWrite(){return OpenNamedPipe(Write);}// const &: const std::string &XXX 输入// *      : std::string *  输出// &      : std::string &  输入输出int WriteNamedPipe(const std::string &in){return write(_fd, in.c_str(), in.size());}int ReadNamedPipe(std::string *out){char buffer[BaseSize];int n = read(_fd, buffer, sizeof(buffer));if (n > 0){buffer[n] = 0;*out = buffer;}return n;}~NamePiped(){if (_id == Creater){int res = unlink(_fifo_path.c_str());if (res != 0)perror("unlink");std::cout << "creater free named pipe" << std::endl;}if (_fd != DefaultFd)close(_fd);}private:const std::string _fifo_path;int _id;int _fd;
};
server.cc
#include "namedPipe.hpp"// 读read
int main()
{// 对于读端而言,如果我们打开文件,但是写还没来,我会阻塞在open调用中,直到对方打开// 也就是相当于进程同步NamePiped fifo(comm_path, Creater);if (fifo.OpenForRead()){std::cout << "server open named pipe done" << std::endl;sleep(3);while (true){std::string message;int n = fifo.ReadNamedPipe(&message);if (n > 0){std::cout << "Client Say> " << message << std::endl;}else if (n == 0){std::cout << "Client quit, Server Too!" << std::endl;break;}else{std::cout << "fifo.ReadNamedPipe Error" << std::endl;break;}}}return 0;
}
client.cc
#include "namedPipe.hpp"// 写Write
int main()
{NamePiped fifo(comm_path, User);if (fifo.OpenForWrite()){std::cout << "client open namd pipe done" << std::endl;while (true){std::cout << "Please Enter> ";std::string message;std::getline(std::cin, message);fifo.WriteNamedPipe(message);}}return 0;
}
  • 首先启动程序,对于读端而言,如果我们打开文件,但是写还没来,我会阻塞在open调用中,直到对方打开
  • 如何关闭写端,读端会读到0,程序会结束

在这里插入图片描述

  • 这次我们先关闭读端,这个时候写端不会立即结束程序,当我们再次输入的时候程序才会退出

在这里插入图片描述

实现日志

Log.hpp
#pragma once#include <iostream>
#include <time.h>
#include <stdarg.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>#define SIZE 1024#define Info 0
#define Debug 1
#define Warning 2
#define Error 3
#define Fatal 4#define Screen 1
#define Onefile 2
#define Classfile 3#define LogFile "log.txt"class Log
{
public:Log(){_printMethod = Screen;_path = "./log/";}// 选择将日志打印到哪void Enable(int method){_printMethod = method;}// 头信息std::string levelToString(int level){switch (level){case Info:return "Info";case Debug:return "Debug";case Warning:return "Warning";case Error:return "Error";case Fatal:return "Fatal";default:return "None";}}// 打印日志void printLog(int level, const std::string &logtxt){switch (_printMethod){case Screen:std::cout << logtxt << std::endl;break;case Onefile:printOneFile(LogFile, logtxt);break;case Classfile:printClassFile(level, logtxt);break;default:break;}}// 单文件打印void printOneFile(const std::string &logname, const std::string &logtxt){std::string _logname = _path + logname;int fd = open(_logname.c_str(), O_WRONLY | O_CREAT | O_APPEND, 0666);if (fd < 0)return;write(fd, logtxt.c_str(), logtxt.size());close(fd);}// 多文件打印void printClassFile(int level, const std::string &logtxt){std::string filename = LogFile;filename += '.';filename += levelToString(level);printOneFile(filename, logtxt);}~Log(){}// 日志格式控制void operator()(int level, const char *format, ...){time_t t = time(nullptr);struct tm *ctime = localtime(&t);char leftbuffer[SIZE];snprintf(leftbuffer, sizeof(leftbuffer), "[%s][%d-%d-%d %d:%d:%d]", levelToString(level).c_str(), ctime->tm_year + 1900, ctime->tm_mon + 1, ctime->tm_mday, ctime->tm_hour, ctime->tm_min, ctime->tm_sec);va_list s;va_start(s, format);char rightbuffer[SIZE];vsnprintf(rightbuffer, sizeof(rightbuffer), format, s);va_end(s);char logtxt[SIZE * 2];snprintf(logtxt, sizeof(logtxt), "%s %s\n", leftbuffer, rightbuffer);printLog(level, logtxt);}private:int _printMethod;std::string _path;
};

共享内存

共享内存原理

在这里插入图片描述

  1. 都是OS做的
  2. OS提供上面的1,2 步骤的系统调用,供用户进程A,B进行调用 (系统调用)
  3. AB,CD,EF,XY----->共享内存在系统中存在多份,供不同个数,不同对进程同时通信
  4. OS注定了要对共享内存进行管理(先描述,再组织),共享内存不是简单的一段内存空间,也要有描述并管理共享内存的数据结构和匹配的算法
  5. 共享内存 = 内存空间(数据) + 共享内存的属性

补充指令集(IPC的指令)

shmget
  • 创建共享内存
int shmget(key_t key, size_t size, int shmflg);
  • 第一个参数是key(后面介绍)
  • 第二个参数创建共享内存的大小(单位是字节)
  • 第三个参数就是位图(下面介绍)

在这里插入图片描述

  • 成功返回共享内存的标识符,失败返回-1,错误码被设置

在这里插入图片描述

  • IPC_CREAT:如果申请的共享内存不存在就创建,存在就获取共享内存并返回
  • IPC_EXCL:单独使用没有意义,只有和IPC_CREAT组合才有意义
  • IPC_CREAT | IPC_EXCL:如果要创建的共享内存不存在就创建,如果存在就出错返回,如果返回成功就意味着这个shm是全新的

在这里插入图片描述


  • 那么如何保证让不同的进程看到同一个共享内存?
  • 怎么保证这个共享内存是存在还是不存在呢?

就是通过 第一个参数key


谈谈key
  1. key是一个数字,这个数字是几,不重要。关键在于必须在内核中具有唯一性,能够让不同的进程进行唯一标识
  2. 第一个进程key通过key创建共享内存,第二个之后的进程, 只要拿着同一个key,就可以和第一个进程看到同一个共享内存了
  3. 对于一个已经创建好的共享内存,key在哪? ----> key在共享内存的描述对象中
  4. 第一次创建的时候,必须有一个key,怎么有?(一会谈)
  5. key 类似之前的路径,都是唯一的

ftok
  • 形成key就使用下面的接口
key_t ftok(const char *pathname, int proj_id);
  • 第一个参数是路径名字符串
  • 第二个参数是项目id

这两个参数由用户自由指定
在这里插入图片描述

那么这个key值能不能由操作系统自动生成,为什么要用户去设置,主要原因是因为操作系统形成了一个key,另一个进程要用这个key,但是我们不知道,所以是由用户约定的,必须由用户层下达到操作系统

key:操作系统内标定的唯一性
shmid:只在你的进程内,用来表示资源的唯一性


ipcs
  • 共享内存的生命周期是随内核的,用户不主动关闭,共享内存会一直存在,除非内核重启或者用户关闭

查看共享内存

ipcs -m

关闭共享内存

  • 这里的shmid是要关闭的共享内存,而不是key,在用户层只能使用shmid,内核层用的key
ipcrm -m shmid
  • 共享内存的大小一般建议是4096的整数倍

shmat
  • 将共享内存挂接到程序地址空间当中
void *shmat(int shmid, const void *shmaddr, int shmflg);
  • 第一个参数是shmid
  • 第二个参数是要挂接到哪个地址上,一般是nullptr
  • 第三个参数是挂接内存的访问权限,默认我们设置为0

返回值:失败返回nullptr,成功返回共享内存的起始地址
在这里插入图片描述

shmdt
  • 从进程的地址空间中分离一个共享内存段
int shmdt(const void *shmaddr);

在这里插入图片描述

shmctl
  • 删除共享内存&&获取共享内存的属性…
int shmctl(int shmid, int cmd, struct shmid_ds *buf);

参数说明:

  • shmid:共享内存段的标识符,通常由 shmget 函数返回。
  • cmd:要执行的操作的命令,可以是以下值之一:
    • IPC_STAT:获取共享内存段的状态,并将结果写入 buf 所指向的 shmid_ds 结构。
    • IPC_SET:设置共享内存段的 shmid_ds 结构中的 shm_perm 字段,通常用于更改权限。
    • IPC_RMID:立即删除共享内存段。注意,只有当共享内存段的引用计数(即附加到它的进程数)为 0 时,该命令才会成功
    • buf:一个指向 shmid_ds 结构的指针,该结构用于传递或接收关于共享内存段的信息。对于 - - IPC_STAT 命令,该结构用于接收信息;对于 IPC_SET 命令,该结构包含要设置的权限信息。

返回值:

如果成功,返回 0。
如果失败,返回 -1,并设置 errno 以指示错误原因。

在这里插入图片描述


  • 共享内存不提供对共享内存的任何保护机制,这会导致数据不一致
  • 共享内存是所有进程IPC,速度最快的,因为共享内存大大减少了数据的拷贝次数!

代码验证(使用共享内存的相关接口)

  • 这里用上前面的log打印日志
Shm.hpp
#ifndef __SHM_HPP__
#define __SHM_HPP__#include <iostream>
#include <string>
#include <cerrno>
#include <cstdio>
#include <cstring>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <unistd.h>#include "Log.hpp"
const int gCreater = 1;
const int gUser = 2;const std::string gpathname = "/root/111/code-exercise/Linux/lesson07";
const int gproj_id = 0x666;
const int gshmSize = 4096;Log log;class Shm
{
public:Shm(const std::string &pathname, int proj_id, int who): _pathname(pathname), _proj_id(proj_id), _who(who), _addrshm(nullptr){_key = GetCommKey(); // 获取key// 通过不同身份创建共享内存,先创建if (_who == gCreater)GetShmUseCreate();else if (_who == gUser)GetShmForUse();std::cout << "shmid: " << _shmid << std::endl;std::cout << "key: " << ToHex(_key) << std::endl;_addrshm = AttachShm(); // 后挂接}// 创建bool GetShmUseCreate(){if (_who == gCreater){_shmid = GetShmHelper(_key, gshmSize, IPC_CREAT | IPC_EXCL | 0666);if (_shmid < 0)return false;log(Info, "GetShmUseCreate share memory success, shmid: %d", _shmid);}return true;}// 使用bool GetShmForUse(){if (_who == gUser){_shmid = GetShmHelper(_key, gshmSize, IPC_CREAT | 0666);if (_shmid < 0)return false;log(Info, "GetShmForUse share memory success, shmid: %d", _shmid);}return true;}std::string ToHex(key_t key){char buffer[128];snprintf(buffer, sizeof(buffer), "%#x", key);return buffer;}~Shm(){if (_who == gCreater){int res = shmctl(_shmid, IPC_RMID, nullptr);if (res < 0){log(Fatal, "shmctl fail! : %s", strerror(errno));exit(3);}}log(Info, "shm remove done ...");}void *Addr(){return _addrshm;}// 将共享内存全部清空void Zero(){if (_addrshm){memset(_addrshm, 0, gshmSize);}}void DebugShm(){// 获取共享内存的属性struct shmid_ds ds;int n = shmctl(_shmid, IPC_STAT, &ds);if (n < 0)return;std::cout << "ds.shm_perm.__key : " << ToHex(ds.shm_perm.__key) << std::endl;std::cout << "ds.shm_nattch: " << ds.shm_nattch << std::endl;}private:// 创建keykey_t GetCommKey(){key_t k = ftok(_pathname.c_str(), _proj_id);if (k < 0){log(Fatal, "ftok, error: %s", strerror(errno));exit(1);}log(Info, "ftok success, key is %#x", k);return k;}int GetShmHelper(key_t key, int size, int flag){int shmid = shmget(key, size, flag); // 获取keyif (shmid < 0){log(Fatal, "create share memory error: %s", strerror(errno));exit(2);}log(Info, "create share memory success, shmid: %d", shmid);std::cout << "create share memory success, key: " << ToHex(_key) << std::endl;return shmid;}std::string RoleToString(int who){if (who == gCreater)return "Creater";else if (who == gUser)return "gUser";elsereturn "None";}void DetachShm(void *shmaddr){if (shmaddr == nullptr)return;shmdt(shmaddr); // 从进程的地址空间中分离一个共享内存段log(Info, "who: %s detach shm...", RoleToString(_who).c_str());}void *AttachShm(){if (_addrshm != nullptr)DetachShm(_addrshm);void *shaddr = shmat(_shmid, nullptr, 0); // 将共享内存挂接到程序地址空间当中if (shaddr == nullptr){log(Fatal, "shmat fail!");}//std::cout << "who: " << RoleToString(_who) << " attach shm..." << std::endl;log(Info, "who: %s attach shm...", RoleToString(_who).c_str());return shaddr;}private:key_t _key;int _shmid;std::string _pathname;int _proj_id;int _who;void *_addrshm;
};#endif
client.cc
#include "Shm.hpp"
#include "Log.hpp"
#include "namedPipe.hpp"int main()
{// 1. 创建共享内存Shm shm(gpathname, gproj_id, gUser);shm.Zero(); // 清空共享内存char *shmaddr = (char *)shm.Addr();// 打印信息shm.DebugShm();// 2. 打开管道NamePiped fifo(comm_path, User);fifo.OpenForWrite();// 当成stringchar ch = 'A';while (ch <= 'Z'){shmaddr[ch - 'A'] = ch;std::string temp = "wakeup";std::cout << "add " << ch << " into Shm, "<< "wakeup reader" << std::endl;fifo.WriteNamedPipe(temp);sleep(2);ch++;}return 0;
}
server.cc
#include "Shm.hpp"
#include "Log.hpp"
#include "namedPipe.hpp"int main()
{Shm shm(gpathname, gproj_id, gCreater);char *shmaddr = (char *)shm.Addr();// 查看信息shm.DebugShm();// 2. 创建管道NamePiped fifo(comm_path, Creater);fifo.OpenForRead();while (true){std::string temp;fifo.ReadNamedPipe(&temp);std::cout << "shm memory content: " << shmaddr << std::endl;}return 0;
}

system V消息队列

  • 消息队列提供了一个从一个进程向另外一个进程发送一块数据的方法

  • 每个数据块都被认为是有一个类型接收者进程接收的数据块可以有不同的类型值
    特性方面:IPC资源必须删除,否则不会自动清除,除非重启,所以system V IPC资源的生命周期随内核

  • 消息队列是由操作系统来提供的

在这里插入图片描述

system V信号量

5个概念:

  1. 多个执行流(进程),能看到的一份资源:共享资源
  2. 被保护起来的资源 —>临界资源同步和互斥,用互斥的方式保护共享资源,临界资源
  3. 互斥:任何时刻只能有一个进程在访问公共资源
  4. 资源:要被程序员访问,资源被访问也就是通过代码来访问(代码 = 访问共享资源的代码(临界区) + 不访问共享资源的代码(非临界区))
  5. 所谓的对共享资源进行保护(临界资源)本质是对共享资源的代码进行保护
  • 这里的信号量也就相当于是一个计数器
  1. 申请计数器成功,就表示具有访问资源的权限了
  2. 申请了计数器资源,我当前访问我要的资源了吗?没有,申请了计数器资源是对资源的预定机制
  3. 计数器可以有效保证进入共享资源的执行流的数量
  4. 所以每一个执行流,想访问共享资源中的一部分资源,不是直接访问,而是先申请计数器

程序员把这个计数器,叫做信号量

进程互斥

  • 由于各进程要求共享资源,而且有些资源需要互斥使用,因此各进程间竞争使用这些资源,进程的这种关系为进程的互斥
  • 系统中某些资源一次只允许一个进程使用,称这样的资源为临界资源或互斥资源。
  • 在进程中涉及到互斥资源的程序段叫临界区

特性方面:

  • IPC资源必须删除,否则不会自动清除,除非重启,所以system V IPC资源的生命周期随内核

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

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

相关文章

【JavaSE】/*运算符—快速总结*/

目录 前言 一、什么是运算符 二、算术运算符 三、增量运算符 四、自增/自减运算符 五、关系运算符 六、逻辑运算符 七、位运算符 八、移位运算符 九、条件运算符 十、运算符的优先级 前言 Java 中的运算符和 C语言 的运算符规则有很多类型的地方&#xff0c;我们只…

视频监控系统中,中心录像服务器的录像文件实际大小和理论值相差很大的问题解决

目录 一、现象描述 二、视频监控的录像文件计算 &#xff08;一&#xff09;计算方法 1、仅视频部分 2、视频和音频部分 3、使用平均码率 &#xff08;二&#xff09;计算工具 1、关注威迪斯特公众号 2、打开“计算容量”的小工具 三、原因分析 &#xff08;一&…

【Win10点击任务栏刷屏,卡死转圈(亲测有效)】

计算机疑难杂症001 Win10点击任务栏刷屏&#xff0c;卡死转圈(亲测有效)1、问题状况2、问题原因3、问题解决 Win10点击任务栏刷屏&#xff0c;卡死转圈(亲测有效) 1、问题状况 在偶然间&#xff0c;发现任务栏点不动了&#xff0c;点击无反应&#xff0c;再多点击几次&#x…

颍川诞生了两个帝王的仲父

伯、仲、叔、季是古代兄弟的长幼排行顺序&#xff0c;《释名释亲属》载&#xff1a;“父之弟曰仲父……仲父之弟曰叔父”。也就是古代称父亲的兄弟为仲父&#xff0c;多用于帝王对宰相重臣的尊称。 历史上最有名的、有正史记载的帝王“仲父”有两位&#xff0c;而且都出自颍川…

nc生成临时凭证配置

nc生成临时凭证配置 要实现的功能&#xff1a; 审批时生成临时凭证弃审时删除临时凭证 前台配置 后台配置 BillReflectorServiceImpl.java package nc.pubimpl.jych.qtsq.voucher;import java.util.ArrayList; import java.util.Collection; import java.util.HashMap; impo…

LeetCode 106.从中序与后序遍历序列构造二叉树

LeetCode 106.从中序与后序遍历序列构造二叉树 1、题目 题目链接&#xff1a;106. 从中序与后序遍历序列构造二叉树 给定两个整数数组 inorder 和 postorder &#xff0c;其中 inorder 是二叉树的中序遍历&#xff0c; postorder 是同一棵树的后序遍历&#xff0c;请你构造并…

Scala编程入门:从零开始的完整教程

目录 引言环境准备创建第一个Scala项目基本语法高阶概念进阶资源结语 引言 Scala是一种强大的、静态类型的、多范式编程语言&#xff0c;它结合了面向对象和函数式编程的特点。本教程将指导您如何从零开始学习Scala&#xff0c;并搭建一个简单的开发环境。让我们开始探索Scala…

搭建一个Xx431?

搭建一个Xx431? 嘿uu们!刚结束了一周六天班感觉如何? 我的状态倒还行,工作生活总能找到乐子,本周整活就是用纸巾和蛋糕托做的油灯,另外想制冷片做个温水冷水可调的杯托,但我还不会搞3d,希望今年能搞起来. 题外话就说到这,这个选题也是因为实际遇到的问题需要这玩意,下班路…

配置Docker对象与管理守护进程

前言&#xff1a;本博客仅作记录学习使用&#xff0c;部分图片出自网络&#xff0c;如有侵犯您的权益&#xff0c;请联系删除 本章节的快速目录导航&#xff1a; 一、配置Docker对象 1.1、Docker对象的标记 1.2、格式化命令和日志的输出 二、示例&#xff1a; 2.1、管理…

【Ubuntu 安装erlang】

apt-get 安装 apt-get install erlang或 源码安装 git clone https://github.com/erlang/otp.git cd otp git checkout maint-25 # current latest stable version ./configure make make install安装完后&#xff0c;验证是否成功 # 命令行输入 erl

分布式事务?哪几种方式实现?一文看懂!

什么是分布式事务 分布式事务是指在分布式系统中涉及到多个数据库或多个应用程序之间的事务处理&#xff0c;这些数据库或应用程序可能分布在不同的物理节点上&#xff0c;甚至可能位于不同的地理位置。在分布式事务中&#xff0c;需要确保所有参与者的事务操作都能够保持一致性…

分布式链路追踪 Zipkin+Sleuth(8)

项目的源码地址 Spring Cloud Alibaba 工程搭建&#xff08;1&#xff09; Spring Cloud Alibaba 工程搭建连接数据库&#xff08;2&#xff09; Spring Cloud Alibaba 集成 nacos 以及整合 Ribbon 与 Feign 实现负载调用&#xff08;3&#xff09; Spring Cloud Alibaba Ribbo…

【姿态解算与滤波算法】

姿态解算 一、主线 姿态表示方式&#xff1a;矩阵表示&#xff0c;轴角表示&#xff0c;欧拉角表示&#xff0c;四元数表示。 惯性测量单元IMU&#xff08;Inertial Measurement Unit&#xff09;&#xff1a;MPU6050芯片&#xff0c;包含陀螺仪和加速度计&#xff0c;分别测…

表面的相似,本质的不同

韩信与韩王信&#xff0c;两个韩信的结局都是被刘邦所杀&#xff0c;似乎结局类似。但是&#xff0c;略加分析&#xff0c;就会发现其中存在本质的区别。 韩信属于必杀。他的王位是要来的&#xff0c;有居功自傲的本意&#xff0c;功高震主而且毫不避讳。而且年轻&#xff0c;…

代码随想录算法训练营第二十五天 | 669. 修剪二叉搜索树、108.将有序数组转换为二叉搜索树、538.把二叉搜索树转换为累加树

669. 修剪二叉搜索树 题目链接/文章讲解&#xff1a; 代码随想录 视频讲解&#xff1a; 你修剪的方式不对&#xff0c;我来给你纠正一下&#xff01;| LeetCode&#xff1a;669. 修剪二叉搜索树_哔哩哔哩_bilibili 解题思路 在上一题的删除二叉树节点中&#xff0c;我们通过在…

2016-2021年全国范围的2.5m分辨率的建筑屋顶数据

一、论文介绍 摘要&#xff1a;大规模且多年的建筑屋顶面积&#xff08;BRA&#xff09;地图对于解决政策决策和可持续发展至关重要。此外&#xff0c;作为人类活动的细粒度指标&#xff0c;BRA可以为城市规划和能源模型提供帮助&#xff0c;为人类福祉带来好处。然而&#xf…

【数组算法】598. 区间加法

给你一个 m x n 的矩阵 M 和一个操作数组 op 。矩阵初始化时所有的单元格都为 0 。ops[i] [ai, bi] 意味着当所有的 0 < x < ai 和 0 < y < bi 时&#xff0c; M[x][y] 应该加 1。 在 执行完所有操作后 &#xff0c;计算并返回 矩阵中最大整数的个数 。 示例 1: …

C语言的打字小游戏

目录 游戏内容 伪代码 main 函数 随机字符串 匹配逻辑 用户决定是否继续或退出游戏 完整代码 从0开始记录我的学习历程&#xff0c;我会尽我所能&#xff0c;写出最最大白话的文章&#xff0c;希望能够帮到你&#xff0c;谢谢。 提示&#xff1a;文章作者为初学者&#xf…

【算法】动态规划之背包DP问题(2024.5.11)

前言&#xff1a; 本系列是学习了董晓老师所讲的知识点做的笔记 董晓算法的个人空间-董晓算法个人主页-哔哩哔哩视频 (bilibili.com) 动态规划系列 【算法】动态规划之线性DP问题-CSDN博客 01背包 步骤&#xff1a; 分析容量j与w[i]的关系&#xff0c;然后分析是否要放…

解决“电脑开机黑屏Explorer进程卡死“问题

今天&#xff0c;给台式机按电源键&#xff0c;进入windows系统时&#xff0c;发现电脑黑屏了&#xff0c;昨天还好好的&#xff0c;怎么今天电脑桌面进不去了&#xff1f;想起Windows XP、Windows 7、Windows 10 、Windows 11等系统&#xff0c;在使用多个文件拷贝时&#xff…