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.mzph.cn/pingmian/10318.shtml

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

相关文章

串口属性中的BM延时计时器问题

如果使用程序修改则需要修改注册表对应位置如下 第一个示例&#xff08;217&#xff09; 第二个示例&#xff08;219&#xff09; 需要注意的事情是修改前必须点查看串口名称&#xff08;例如上图是com5&#xff09; 程序修改&#xff1a; 有没有办法以编程方式更改USB <…

【力扣】63.不同路径 II

原题链接&#xff1a;. - 力扣&#xff08;LeetCode&#xff09; 目录 1.题目描述 2.思路分析 3.代码实现 1.题目描述 一个机器人位于一个 m x n 网格的左上角 &#xff08;起始点在下图中标记为 “Start” &#xff09;。 机器人每次只能向下或者向右移动一步。机器人试…

uniapp管理后台编写,基于uniadmin和vue3实现uniapp小程序的管理后台

一&#xff0c;创建uniAdmin项目 打开开发者工具Hbuilder,然后点击左上角的文件&#xff0c;点新建&#xff0c;点项目。如下图。 选择uniadmin&#xff0c;编写项目名&#xff0c;然后使用vue3 记得选用阿里云服务器&#xff0c;因为最便宜 点击创建&#xff0c;等待项目创…

Codeforces Round 605 (Div. 3) A~D

本人水平不高&#xff0c;开这个专栏主要是督促自己补题&#xff0c;有些题对目前的我来说还比较难&#xff0c;还补不动&#xff0c;等以后能力上来了再补。。。 原题链接&#xff1a;Dashboard - Codeforces Round 605 (Div. 3) - Codeforces 目录 A. Three Friends B. Sn…

【Linux:lesson1】的基本指令

&#x1f381;个人主页&#xff1a;我们的五年 &#x1f50d;系列专栏&#xff1a;Linux课程学习 &#x1f337;追光的人&#xff0c;终会万丈光芒 &#x1f389;欢迎大家点赞&#x1f44d;评论&#x1f4dd;收藏⭐文章 目录 &#x1f697;打开Xshell&#xff0c;登陆root…

初识java--javaSE(3)--方法,递归,数组,

文章目录 一 方法的使用1.1 什么是方法&#xff1f;main方法注意事项 1.2 方法的调用嵌套调用在方法调用时形参与实参的关系&#xff1a; 1.3 方法的重载方法重载的意义&#xff1f;总结方法重载&#xff1a;方法签名&#xff1a; 二 递归什么是递归&#xff1f;递归的精髓&…

蛋糕店做配送小程序的作用是什么

蛋糕烘焙除了生日需要&#xff0c;对喜吃之人来说往往复购率较高&#xff0c;除线下实体店经营外&#xff0c;更多的商家选择线上多种方式获客转化、持续提高生意营收&#xff0c;而除了进驻第三方平台外&#xff0c;构建品牌私域自营店铺也同样重要。 运用【雨科】平台搭建蛋…

ABAP跨client的RFC调用

1、SM59配置连接 2、创建需要调用的函数&#xff0c;ZGET_TM_LIST&#xff0c;开启远程启用模块 3、新建调用程序 DATA:L_MSG TYPE C,LSH(30) TYPE C. DATA:IT_ZSTM_LIST TYPE STANDARD TABLE OF ZSTM_LIST WITH HEADER LINE.CALL FUNCTION ZGET_TM_LIST DESTINATION ZTEST_R…

【回溯 网格 状态压缩】52. N 皇后 II

本文涉及知识点 回溯 网格 状态压缩 LeetCode52. N 皇后 II n 皇后问题 研究的是如何将 n 个皇后放置在 n n 的棋盘上&#xff0c;并且使皇后彼此之间不能相互攻击。 给你一个整数 n &#xff0c;返回 n 皇后问题 不同的解决方案的数量。 示例 1&#xff1a; 输入&#x…

python3如何安装bs4

在python官网找到beautifulsoup模块的下载页面&#xff0c;点击"downloap"将该模块的安装包下载到本地。 将该安装包解压&#xff0c;然后在打开cmd&#xff0c;并通过cmd进入到该安装包解压后的文件夹目录下。 在该文件目录下输入"python install setup.py&quo…

信息系统架构模型_3.企业数据交换总线

1.企业数据交换总线 实践中&#xff0c;还有一种较常用的架构&#xff0c;即企业数据交换总线&#xff0c;即不同的企业应用之间进行信息交换的公共通道&#xff0c;如图1所示。 图1 企业数据交换总线架构 这种架构在大型企业不同应用系统进行信息交换时使用较普遍&am…

前端笔记-day05

文章目录 01-结构伪类选择器02-结构伪类选择器-公式用法03-伪元素选择器04-盒子模型-组成05-盒子模型-边框线06-盒子模型-单方向边框线07-盒子模型-内边距08-盒子模型-padding多值写法09-盒子模型-尺寸计算10-盒子模型-版心居中11-清除默认样式12-元素溢出overflow13-外边距合并…

嵌入式基础课程配套电机FOC伺服电机开发板AT32F403磁编码IMU姿态

嵌入式基础课程配套电机FOC伺服电机开发板AT32F403磁编码IMU姿态 带你入门嵌入式有二十多年开发经验的老技骨做技术支持整个开发包硬件包括电机2205&#xff0c;支持12V到24V宽输入&#xff0c;配套12V2A电源。包装原理图和PCB嵌入式软件嵌入式基础课程 带你入门嵌入式 电机FO…

即插即用篇 | YOLOv8引入局部自注意力 HaloAttention | 为参数高效的视觉主干网络扩展局部自注意力

本改进已集成到 YOLOv8-Magic 框架。 我们提出了Axial Transformers,这是一个基于自注意力的自回归模型,用于图像和其他组织为高维张量的数据。现有的自回归模型要么因高维数据的计算资源需求过大而受到限制,要么为了减少资源需求而在分布表达性或实现的便捷性上做出妥协。相…

嵌入式C语言高级教程:实现基于STM32的无人机飞控系统

无人机飞控系统是无人机的大脑&#xff0c;负责处理来自各种传感器的数据并控制无人机的飞行。本教程将指导如何在STM32微控制器上实现一个基础的无人机飞控系统。 一、开发环境准备 硬件要求 微控制器&#xff1a;STM32F405RGT6&#xff0c;因其高性能和大量的输入输出接口…

机器学习算法应用——K近邻分类器(KNN)

K近邻分类器&#xff08;KNN&#xff09;&#xff08;4-2&#xff09; K近邻分类器&#xff08;K-Nearest Neighbor&#xff0c;简称KNN&#xff09;是一种基本的机器学习分类算法。它的工作原理是&#xff1a;在特征空间中&#xff0c;如果一个样本在特征空间中的K个最相邻的样…

交通数据三维可视化呈现与可视化分析系统开发(附程序源码)

目录 01 系统介绍 02 功能介绍 文件管理功能 模型研究 可视化分析功能 今天分享一套“交通数据三维可视化呈现与可视化分析系统”&#xff0c;并开放程序源代码下载&#xff0c;内容涉及开源空间数据库的使用、三维引擎的二次开发、矢量和栅格数据管理、交通流量分析模型框…

【18-Ⅱ】Head First Java 学习笔记

HeadFirst Java 本人有C语言基础&#xff0c;通过阅读Java廖雪峰网站&#xff0c;简单速成了java&#xff0c;但对其中一些入门概念有所疏漏&#xff0c;阅读本书以弥补。 第一章 Java入门 第二章 面向对象 第三章 变量 第四章 方法操作实例变量 第五章 程序实战 第六章 Java…

Qt | QSpinBox 类 QDoubleSpinBox 类(微调框)

01、QSpinBox 类 1、QSpinBox类是 QAbstractSpinBox 类的直接子类和具体实现, 2、QSpinBox 类被设计用于处理整数和离散值集合,对于浮点值使用 QDoubleSpinBox 类实现。 3、QSpinBox 默认只支持整数值,但可通过其内部的成员函数进行扩展,以支持使用不同的 字符串。 02…

ROS2 工作空间

文章目录 ROS2 工作空间创建工作空间自动安装依赖编译工作空间设置环境变量参考链接 ROS2 工作空间 工作空间可以简单理解为工程目录。 ROS 系统中一个典型的工作空间结构如图所示&#xff1a; dev_ws&#xff1a; 根目录&#xff0c;里面会有四个子目录&#xff08;子空间&a…