【Linux】进程间通信_4

简介: 【Linux】进程间通信_4

七、进程间通信

1. 进程间通信分类

systeam V共享内存

进程间通信的本质就是让不同进程看到同一份资源。而systeam V是通过让不同的进程经过页表映射到同一块内存空间(操作系统完成的)。

我们申请的共享内存,如果进程结束了,但共享内存并不会释放,需要我们手动释放。管道文件的生命周期是随进程的,但是共享内存的生命周期是随内核的。

使用 ipcs -m 命令可以查看系统中我们创建的共享内存数量。使用 ipcrm -m 和要删除的共享内存的 shmid 即可删除指定共享内存。

这里我们来实现一下共享内存模式的进程间通信:

Makefile

.PHONY:all
all:shm_server shm_client
shm_server:ShmServer.cc
  g++ -o $@ $^ -std=c++11
shm_client:ShmClient.cc
  g++ -o $@ $^ -std=c++11
.PHONY:clean
clean:
  rm -f shm_server shm_client

Comm.hpp

#pragma once
#include <iostream>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/types.h>
#include <cerrno>
#include <cstring>
#include <cstdlib>
#include <string>
#include <unistd.h>
using namespace std;
const char* pathname = "/home/student";
const int proj_id = 0x66;
const int defaultsize = 4096;
// 将key转换为16进制
string ToHex(key_t k)
{
    char buffer[1024];
    snprintf(buffer, sizeof(buffer), "0x%x", k);
    return buffer;
}
// 生成共享内存的key
key_t GetShmKeyOrDie()
{
    key_t k = ftok(pathname, proj_id);
    if (k < 0)
    {
        cerr << "ftok error, errno : " << errno << ", errno string : " << strerror(errno) << endl;
        exit(1);
    }
    return k;
}
// 根据key创建共享内存
int CreateShmOrDie(key_t key, int size, int flag)
{
    int shmid = shmget(key, size, flag);
    if (shmid < 0)
    {
        cerr << "shmget error, errno : " << errno << ", errno string : " << strerror(errno) << endl;
        exit(2);
    }
    return shmid;
}
int CreateShm(key_t key, int size)
{
    return CreateShmOrDie(key, size, IPC_CREAT | IPC_EXCL | 0666);
}
int GetShm(key_t key, int size)
{
    return CreateShmOrDie(key, size, IPC_CREAT);
}
// 删除共享内存
void DeleteShm(int shmid)
{
    int n = shmctl(shmid, IPC_RMID, nullptr);
    if (n < 0)
    {
        cerr << "shmctl error, errno : " << errno << ", errno string : " << strerror(errno) << endl;
    }
    else
    {
        cout << "shmctl delete shm success, shmid : " << shmid << endl;
    }
}
// 将共享内存附加到进程的地址空间,实现映射关系
void* ShmAttach(int shmid)
{
    void* addr = shmat(shmid, nullptr, 0);
    if (addr == (void*)-1)
    {
        cerr << "shmat error, errno : " << errno << ", errno string : " << strerror(errno) << endl;
        return nullptr;
    }
    return addr;
}
// 解除共享内存的映射关系
void ShmDetach(void* addr)
{
    int n = shmdt(addr);
    if (n < 0)
    {
        cerr << "shmdt error, errno : " << errno << ", errno string : " << strerror(errno) << endl;
    }
}

Fifo.hpp

#ifndef __COMM_HPP__
#define __COMM_HPP__
#include <iostream>
#include <string>
#include <sys/types.h>
#include <sys/stat.h>
#include <cerrno>
#include <cstring>
#include <unistd.h>
#include <fcntl.h>
#include <cassert>
using namespace std;
#define Mode 0666
#define Path "./fifo"
// 命名管道类
class Fifo
{
public:
    Fifo(const string& path = Path)
    : _path(path)
    {
        umask(0);
        // 创建命名管道
        int n = mkfifo(_path.c_str(), Mode);
        if (n == 0)
        {
            cout << "mkfifo success" << endl;
        }
        else
        {
            cout << "mkfifo failed, error : " << errno << "errstring : " << strerror(errno) << endl;
        }
    }
    ~Fifo()
    {
        // 删除命名管道
        int n = unlink(_path.c_str());
        if (n == 0)
        {
            cout << "remove " << _path << " success" << endl;
        }
        else
        {
            cout << "remove failed, error : " << errno << "errstring : " << strerror(errno) << endl;
        }
    }
private:
    // 文件路径 + 文件名
    string _path;
};
class Sync
{
public:
    Sync()
    :_rfd(-1)
    ,_wfd(-1)
    {}
    void OpenReadOrDie()
    {
        _rfd = open(Path, O_RDONLY);
        if (_rfd < 0) exit(1);
    }
    void OpenWriteOrDie()
    {
        _wfd = open(Path, O_WRONLY);
        if (_wfd < 0) exit(1);
    }
    bool Wait()
    {
        bool ret = true;
        uint32_t c = 0;
        ssize_t n = read(_rfd, &c, sizeof(uint32_t));
        if (n == sizeof(uint32_t))
        {
            cout << "wait for server" << endl;
        }
        else if (n == 0)
        {
            ret = false;
        }
        else
        {
            return false;
        }
        return ret;
    }
    void WakeUp()
    {
        uint32_t c = 0;
        ssize_t n = write(_wfd, &c, sizeof(uint32_t));
        assert(n == sizeof(uint32_t));
        cout << "wakeup server" << endl;
    }
    ~Sync() {}
private:
    int _rfd;
    int _wfd;
};
#endif

ShmServer

#include "Comm.hpp"
#include "Fifo.hpp"
int main()
{
    // 生成一个key
    key_t key = GetShmKeyOrDie();
    // 生成共享内存
    int shmid = CreateShm(key, defaultsize);
    // 将共享内存和进程进行关联(挂接)
    char *addr = (char*)ShmAttach(shmid);
    // 引入管道
    Fifo fifo;
    Sync syn;
    syn.OpenReadOrDie();
    // 循环读取共享内存
    while(1)
    {
        if (!syn.Wait()) break;
        cout << "Shm content: " << addr << endl;
    }
    // 去关联共享内存
    ShmDetach(addr);
    // 删除共享内存
    DeleteShm(shmid);
    return 0;
}

ShmClient

#include "Comm.hpp"
#include "Fifo.hpp"
int main()
{
    key_t key = GetShmKeyOrDie();
    int shmid = GetShm(key, defaultsize);
    char *addr = (char*)ShmAttach(shmid);
    Sync syn;
    syn.OpenWriteOrDie();
    memset(addr, 0, defaultsize);
    for (int c = 'A'; c <= 'Z'; c++)
    {
        addr[c - 'A'] = c;
        sleep(1);
        syn.WakeUp();
    }
    ShmDetach(addr);
    return 0;
}

结果:

消息队列

消息队列提供了一个从一个进程向另外一个进程发送一块数据的方法。每个数据块都被认为是有一个类型,接收者进程接收的数据块可以有不同的类型值。IPC资源必须删除,否则不会自动清除,除非重启,所以system V IPC资源的生命周期随内核。

消息队列的本质就是操作系统在内核维护了多个数据块队列,不同进程根据数据怪的标识来通信。

信号量

信号量本质是一个计数器,描述临界资源数量的计数器。进程访问临界资源本质就是申请信号量,申请成功才能访问临界资源,否者只能等待申请。所有的进程访问临界资源,都需要申请信号量,所有的进程都必须要看到同一个信号量,说明信号量本身就是一个共享资源。

信号量的申请操作称为 P操作 ,释放操作称为 V操作

在命令行使用 ipcs -s 即可查询信号量, ipcrm -s 加信号量标识符 即可删除指定信号量。


未完待续

目录
相关文章
|
2月前
|
算法 Linux 调度
深入理解Linux操作系统的进程管理
本文旨在探讨Linux操作系统中的进程管理机制,包括进程的创建、执行、调度和终止等环节。通过对Linux内核中相关模块的分析,揭示其高效的进程管理策略,为开发者提供优化程序性能和资源利用率的参考。
110 1
|
4天前
|
存储 网络协议 Linux
【Linux】进程IO|系统调用|open|write|文件描述符fd|封装|理解一切皆文件
本文详细介绍了Linux中的进程IO与系统调用,包括 `open`、`write`、`read`和 `close`函数及其用法,解释了文件描述符(fd)的概念,并深入探讨了Linux中的“一切皆文件”思想。这种设计极大地简化了系统编程,使得处理不同类型的IO设备变得更加一致和简单。通过本文的学习,您应该能够更好地理解和应用Linux中的进程IO操作,提高系统编程的效率和能力。
50 34
|
8天前
|
消息中间件 Linux C++
c++ linux通过实现独立进程之间的通信和传递字符串 demo
的进程间通信机制,适用于父子进程之间的数据传输。希望本文能帮助您更好地理解和应用Linux管道,提升开发效率。 在实际开发中,除了管道,还可以根据具体需求选择消息队列、共享内存、套接字等其他进程间通信方
40 16
|
1月前
|
消息中间件 Linux
Linux:进程间通信(共享内存详细讲解以及小项目使用和相关指令、消息队列、信号量)
通过上述讲解和代码示例,您可以理解和实现Linux系统中的进程间通信机制,包括共享内存、消息队列和信号量。这些机制在实际开发中非常重要,能够提高系统的并发处理能力和数据通信效率。希望本文能为您的学习和开发提供实用的指导和帮助。
119 20
|
2月前
|
存储 监控 Linux
嵌入式Linux系统编程 — 5.3 times、clock函数获取进程时间
在嵌入式Linux系统编程中,`times`和 `clock`函数是获取进程时间的两个重要工具。`times`函数提供了更详细的进程和子进程时间信息,而 `clock`函数则提供了更简单的处理器时间获取方法。根据具体需求选择合适的函数,可以更有效地进行性能分析和资源管理。通过本文的介绍,希望能帮助您更好地理解和使用这两个函数,提高嵌入式系统编程的效率和效果。
121 13
|
2月前
|
SQL 运维 监控
南大通用GBase 8a MPP Cluster Linux端SQL进程监控工具
南大通用GBase 8a MPP Cluster Linux端SQL进程监控工具
|
3月前
|
缓存 监控 Linux
linux进程管理万字详解!!!
本文档介绍了Linux系统中进程管理、系统负载监控、内存监控和磁盘监控的基本概念和常用命令。主要内容包括: 1. **进程管理**: - **进程介绍**:程序与进程的关系、进程的生命周期、查看进程号和父进程号的方法。 - **进程监控命令**:`ps`、`pstree`、`pidof`、`top`、`htop`、`lsof`等命令的使用方法和案例。 - **进程管理命令**:控制信号、`kill`、`pkill`、`killall`、前台和后台运行、`screen`、`nohup`等命令的使用方法和案例。
220 4
linux进程管理万字详解!!!
|
2月前
|
运维 监控 Linux
Linux操作系统的守护进程与服务管理深度剖析####
本文作为一篇技术性文章,旨在深入探讨Linux操作系统中守护进程与服务管理的机制、工具及实践策略。不同于传统的摘要概述,本文将以“守护进程的生命周期”为核心线索,串联起Linux服务管理的各个方面,从守护进程的定义与特性出发,逐步深入到Systemd的工作原理、服务单元文件编写、服务状态管理以及故障排查技巧,为读者呈现一幅Linux服务管理的全景图。 ####
|
3月前
|
缓存 算法 Linux
Linux内核的心脏:深入理解进程调度器
本文探讨了Linux操作系统中至关重要的组成部分——进程调度器。通过分析其工作原理、调度算法以及在不同场景下的表现,揭示它是如何高效管理CPU资源,确保系统响应性和公平性的。本文旨在为读者提供一个清晰的视图,了解在多任务环境下,Linux是如何智能地分配处理器时间给各个进程的。
|
3月前
|
存储 运维 监控
深入Linux基础:文件系统与进程管理详解
深入Linux基础:文件系统与进程管理详解
107 8