菜鸟程序员利用共享内存实现进程间通信

上篇文章介绍了进程间的FIFO通信,FIFO通信属于有名管道通信,其能够用于任何进程间的数据通信。

今天介绍第三种进程通信方式—共享内存

共享内存的概念

共享内存(share memory)是一种最为高效的进程间通信方式,是因为进程能够直接对内存进行读写,且不需要进行数据的保存与复制。

为了实现在多个进程间高效的数据通信,linux内核特地留下一块内存区,该内存区能够被需要的进程映射到自身的内存空间。因此,进程便能够直接对这块内存区进行读写操作。

共享内存的实现进程通信的原理图如下:
在这里插入图片描述

共享内存的实现步骤

共享内存的实现较为简单,一共分为两个步骤:

创建共享内存。通过函数shmget()从内存中获取一块共享内存区域,该函数返回值为共享内存的ID。

映射共享内存。通过函数shmat()将上一步获取的共享内存映射到具体的内存空间。

NOTE:先创建共享内存,再将共享内存映射到每个进程中。

下图为一个简单的共享内存实现流程图:

在这里插入图片描述

共享内存的代码实现

下面是共享内存的代码实现(已在linux环境下编译通过):

//文件名shm.c
#include<unistd.h>
#include<sys/types.h>
#include<sys/wait.h>
#include<stdio.h>
#include<stdlib.h>
#include<sys/ipc.h>
#include<sys/shm.h>
#include<string.h>

#define MAX_BUFFER_LEN 1024

int main()
{
  pid_t pid;
  int shm_id;
  char* shm_addr;
  char flag[] = "WROTE";
  char buff[MAX_BUFFER_LEN];
  int i = 0;

  memset(buff, 0, sizeof(buff));
  shm_id = shmget(IPC_PRIVATE, MAX_BUFFER_LEN, 0666);//创建共享内存文件,并设置文件掩码
  if(shm_id == -1)
  {
    printf("creat share memory fail.\n");
    return 0;
  }
  else
  {
    printf("creat share memeory: %d\n", shm_id);
  }

  system("ipcs -m");

  pid = fork();//创建子进程
  if(pid == -1)
  {
    printf("creat child process fail.\n");
    exit(1);
  }
  else if(pid == 0)//子进程进入该分支
  {
    printf("this is child process.\n");
    shm_addr = shmat(shm_id, 0, 0);//将共享内存映射到子进程
    if(shm_addr == (void*)-1)
    {
      perror("child: shmat");
      exit(1);
    }
    else
    {
      printf("child: attach shared-memory: %p\n", shm_addr);
    }
    system("ipcs -m");

    while((strncmp(shm_addr, flag, strlen(flag))) && (i <= 10))
    {
      printf("child: wait for emable data..\n");
      sleep(5);
      i++;
    }

    strcpy(buff, shm_addr+strlen(flag));
    printf("child: share-memory: %s\n", buff);

    if(shmdt(shm_addr) == (void*)-1)
    {
      perror("shmat");
      exit(1);
    }
    else
    {
      printf("child: deattach shared-memory.\n");
    }

    system("ipcs -m");
    if(shmctl(shm_id, IPC_RMID, NULL) == -1)
    {
      perror("child: shmctl(IPC_RMID)\n");
      exit(1);
    }
    else
    {
      printf("delete shared-memory.\n");
    }
    system("ipcs -m");
    exit(0);
  }
  else//父进程进入该分支
  {
    printf("this is parent process.\n");
    shm_addr = shmat(shm_id, 0, 0);//将共享内存映射到父进程
    if(shm_id == -1)
    {
      perror("creat share memory");
      exit(1);
    }
    else
    {
      printf("parent process share memory address: %p\n", shm_addr);
    }
    sleep(1);
    printf("\ninput some string:\n");
    fgets(buff, MAX_BUFFER_LEN, stdin);
    strncpy(shm_addr+strlen(flag), buff, strlen(buff));
    strncpy(shm_addr, flag, strlen(flag));

    if(shmdt(shm_addr) == -1)
    {
      perror("parent: shmdt");
      exit(1);
    }
    else
    {
      printf("parent: deattach share memory\n");
    }
    system("ipcs -m");

    waitpid(pid, NULL, 0);
    printf("finish\n");
  }
  exit(0);
  return 0;
}
//编译命令为:gcc shm.c -o shm
//运行命令为:./shm

下图是代码运行结果:
在这里插入图片描述

总结

共享内存是一种很高效的进程间通信方式,其原理非常简单,且易于用代码实现,感兴趣的朋友可以自己动手实践一下。

共享内存通信虽说简单,但在实际开发项目应用并不太广泛。下篇文章将介绍一种广泛使用的进程间通信方式—消息队列。

ps: 欢迎关注我的公众号**[酷酷的coder]**,分享转行菜鸟程序员成长过程汇总的烦恼和反思.
在这里插入图片描述

  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个基于信号和共享内存的守护进程同步写一次收一次的C语言代码示例: ```c #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <signal.h> #include <sys/ipc.h> #include <sys/shm.h> #define SHM_KEY 1234 volatile sig_atomic_t flag = 0; void sig_handler(int sig) { flag = 1; } int main() { int shmid; char *shmaddr; struct sigaction sa; // 创建共享内存 shmid = shmget(SHM_KEY, 1024, IPC_CREAT | 0666); if (shmid == -1) { perror("shmget"); exit(EXIT_FAILURE); } // 连接共享内存 shmaddr = shmat(shmid, NULL, 0); if (shmaddr == (void *)-1) { perror("shmat"); exit(EXIT_FAILURE); } // 注册信号处理函数 sa.sa_handler = sig_handler; sa.sa_flags = 0; sigemptyset(&sa.sa_mask); if (sigaction(SIGUSR1, &sa, NULL) == -1) { perror("sigaction"); exit(EXIT_FAILURE); } // 创建子进程 pid_t pid = fork(); if (pid == -1) { perror("fork"); exit(EXIT_FAILURE); } if (pid == 0) { // 子进程 while (1) { pause(); // 等待信号 printf("子进程收到信号\n"); printf("子进程读取共享内存中的数据:%s\n", shmaddr); } } else { // 父进程 while (1) { printf("请输入要写入共享内存的数据:"); fgets(shmaddr, 1024, stdin); kill(pid, SIGUSR1); // 发送信号通知子进程 while (!flag) {} // 等待子进程读取完共享内存 flag = 0; } } // 分离共享内存 if (shmdt(shmaddr) == -1) { perror("shmdt"); exit(EXIT_FAILURE); } // 删除共享内存 if (shmctl(shmid, IPC_RMID, NULL) == -1) { perror("shmctl"); exit(EXIT_FAILURE); } return 0; } ``` 该程序先创建一个共享内存和一个信号处理函数,然后通过fork()函数创建一个子进程。父进程不断从标准输入读取数据,写入共享内存,并向子进程发送一个SIGUSR1信号。子进程等待信号,收到信号后从共享内存中读取数据并打印。父进程等待子进程读取完共享内存后再写入新数据。程序结束前删除共享内存

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值