例子1:两个进程通过共享内存通信,一个进程向共享内存中写入数据,另一个进程从共享内存中读出数据

    文件1 创建进程1,实现功能,打印共享内存中的数据

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <errno.h>
#include <string.h> int main(int argc, const char *argv[])
{
key_t key;
int shmid;
char *p = NULL; // key_t ftok(const char *pathname, int proj_id);;
key = ftok("./app",'b'); //获取唯一的 key 值,
if(key < )
{
perror("fail ftok ");
exit();
} shmid = shmget(key, , IPC_CREAT|IPC_EXCL|);//创建/打开共享内存,返回id根据id映射
if(shmid < )
{
if(errno == EEXIST)//文件存在时,直接打开文件获取shmid
{
printf("file eexist");
shmid = shmget(key,,);
}
else
{
perror("shmget fail ");
exit();
}
}
p = (char *)shmat(shmid,NULL,);//映射,返回地址,根据地址操作
if( p == (char *)(-) )
{
perror("shmat fail ");
exit();
} while()
{
sleep();
printf("P:%s",p);
if(strstr(p,"quit") != NULL) //接收到 quit 结束循环
{
break;
}
}
shmdt(p);//解除映射 //函数原型 int shmctl(int shmid, int cmd, struct shmid_ds *buf);
shmctl(shmid,IPC_RMID,NULL); //删除
return ;
}

  文件2  创建进程2 实现功能,获取终端输入的数据写到共享内存中,这两个进程需要同时启动才可以实现通信

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <errno.h>
#include <string.h> int main(int argc, const char *argv[])
{
key_t key;
int shmid;
char *p = NULL;
// key_t ftok(const char *pathname, int proj_id);;
key = ftok("./app",'b'); //创建key值,
if(key < )
{
perror("fail ftok ");
exit();
}
shmid = shmget(key,,IPC_CREAT|IPC_EXCL|);//创建/打开共享内存,返回id根据id映射
if(shmid < )
{
if(errno == EEXIST)//文件存在时,直接打开文件获取shmid
{
printf("file eexist");
shmid = shmget(key,,); //共享内存存在时,直接打开
}
else
{
perror("shmget fail ");
exit();
}
}
p = (char *)shmat(shmid,NULL,);//映射,返回地址,根据地址操作
if( p == (char *)(-) )
{
perror("shmat fail ");
exit();
}
while() //接收到 quit 结束循环
{
//read 时p中 内容不会清空
//read(0,p,10);//终端读数据,写入p指向的空间,读取数据时不会清空
fgets(p,,stdin);
if(strstr(p,"quit") != NULL)
{
break;
}
}
shmdt(p);//解除映射 //函数原型 int shmctl(int shmid, int cmd, struct shmid_ds *buf);
shmctl(shmid,IPC_RMID,NULL); //删除
return ;
}

测试:在进程2终端下输入数据,会在进程1 的终端下打印出来,但是一直打印(输入quit退出循环)

 

例子2 :使用信号灯集访问临界资源(共享内存)处理上面的一直循环打印的问题

通过创建包含 2 个信号灯的灯集, 分别控制两个进程访问共享内存  ( 2值信号灯 )

文件 1  进程1 通过终端输入数据写入共享内存, 两个信号灯,先允许通过终端 输入数据进程操作

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <errno.h>
#include <string.h>
#include <sys/sem.h> int semid; //信号灯集id
union semnum
{
int val; //信号灯值
}; union semnum mynum; struct sembuf mybuf; //定义操作信号灯的结构,PV操作,哪个灯 //参数 信号灯集id 和 是哪个信号灯
void sem_p(int semid, unsigned short num) //P操作函数
{
mybuf.sem_num = num; //第一个信号灯,(信号灯编号)
mybuf.sem_op = -; //进行P操作, 为 1 时表示V操作
mybuf.sem_flg = ; //阻塞,表示semop函数的操作是阻塞的,直到成功为止
semop(semid, &mybuf, ); //最后一个参数,表示操作信号灯的个数
} //参数 信号灯集id 和 是哪个信号灯
void sem_v(int semid, unsigned short num) //V操作函数
{
mybuf.sem_num = num;
mybuf.sem_op = ; //1 表示V 操作
mybuf.sem_flg = ; //阻塞,表示semop函数的操作是阻塞的,直到操作成功为止
semop(semid, &mybuf, ); //操作的 mybuf 全部变量信号灯集
} int main(int argc, const char *argv[])
{
key_t key;
int shmid;
char *p = NULL; // key_t ftok(const char *pathname, int proj_id);;
key = ftok("./app",'b'); //创建key值,
if(key < )
{
perror("fail ftok ");
exit();
} //IPC_EXCL | IPC_CREAT 信号灯不存在就创建
semid = semget(key, , IPC_CREAT|IPC_EXCL|); //创建信号灯,IPC_EXCL 问信号灯存不存在
if(semid < )
{
if(errno == EEXIST)//存在时,只需要打开即可
{
semid = semget(key,,); //打开信号灯
}
else
{
perror("semget fail ");
       exit(1);
}
}
else
{
mynum.val = ; //设置信号灯值
semctl(semid,,SETVAL,mynum); //初始化一个信号灯 mynum.val = ;
semctl(semid,,SETVAL,mynum);
} shmid = shmget(key,,IPC_CREAT|IPC_EXCL|);//创建/打开共享内存,返回id根据id映射
if(shmid < )
{
if(errno == EEXIST)//文件存在时,直接打开文件获取shmid
{
printf("file eexist");
shmid = shmget(key,,); // 直接打开共享内存
}
else
{
perror("shmget fail ");
exit();
}
} p = (char *)shmat(shmid,NULL,);//映射,返回地址,根据地址操作
if( p == (char *)(-) ) //判断共享内存是否正确
{
perror("shmat fail ");
exit();
}
while()
{
sem_p(semid, ); // P 操作
//read 时 p 指向的共享内存中内容不会清空
//read(0,p,10);//终端读数据,写入p指向的空间,读取数据时不会清空
fgets(p,,stdin);
sem_v(semid, ); // V 操作
if(strstr(p,"quit") != NULL) //当输入 "quit" 退出循环
{
break;
}
}
shmdt(p);//解除映射 //int shmctl(int shmid, int cmd, struct shmid_ds *buf);
shmctl(shmid,IPC_RMID,NULL); //删除共享内存
semctl(semid,,IPC_RMID);//删除信号灯
return ;
}

文件 2  进程2 读取共享内存中的数据, 通过信号灯控制, 当写操作完成时读取一次, 然后等待下次写入

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <errno.h>
#include <string.h>
#include <sys/sem.h> int semid; //信号灯集id
union semnum
{
int val; //信号灯值
}; union semnum mynum; struct sembuf mybuf; //定义操作信号灯的结构 //参数 信号灯集id 和 是哪个信号灯
void sem_p(int semid, unsigned short num) //P操作函数
{
mybuf.sem_num = num; //第一个信号灯,(信号灯编号)
mybuf.sem_op = -; //进行P操作, 为 1 时表示V操作
mybuf.sem_flg = ; //阻塞,表示semop函数操作是阻塞的,直到成功为止
  //函数semop是对信号灯进行PV操作的函数,通过上面的结构体,参数,确定是P操作还是V操作,确定是对哪个信号灯的PV 操作,simid表示操作的是那个信号灯集
semop(semid, &mybuf, ); //最后一个参数,表示操作信号灯的个数
} //参数 信号灯集id 和 是哪个信号灯
void sem_v(int semid, unsigned short num) //V操作函数
{
mybuf.sem_num = num;
mybuf.sem_op = ; //1 表示V 操作
mybuf.sem_flg = ; //阻塞,表示semop函数操作是阻塞的,直到成功为止
semop(semid, &mybuf, ); //操作的 mybuf 全部变量信号灯集
} int main(int argc, const char *argv[])
{
key_t key;
int shmid;
char *p = NULL; // key_t ftok(const char *pathname, int proj_id);;
key = ftok("./app",'b'); //创建key值,
if(key < )
{
perror("fail ftok ");
exit();
} //IPC_EXCL | IPC_CREAT 信号灯不存在就创建
semid = semget(key, , IPC_CREAT|IPC_EXCL|); //创建信号灯,IPC_EXCL 问信号灯存不存在
if(semid < )
{
if(errno == EEXIST)//存在时,只需要打开即可
{
semid = semget(key,,); //打开信号灯
}
else
{
perror("semget fail ");
}
}
else
{
mynum.val = ; //设置信号灯值
semctl(semid,,SETVAL,mynum); //初始化一个信号灯 mynum.val = ;
semctl(semid,,SETVAL,mynum);
} shmid = shmget(key,,IPC_CREAT|IPC_EXCL|);//创建/打开共享内存,返回id根据id映射
if(shmid < )
{
if(errno == EEXIST)//文件存在时,直接打开文件获取shmid
{
printf("file eexist");
shmid = shmget(key,,);
}
else
{
perror("shmget fail ");
exit();
}
}
p = (char *)shmat(shmid,NULL,);//映射,返回地址,根据地址操作
if( p == (char *)(-) )
{
perror("shmat fail ");
exit();
}
while()
{
sleep();
sem_p(semid, );
printf("P:%s",p);
sem_v(semid, );
// write(0,p,10);//向终端写数据,把p指向的空间的内容
if(strstr(p,"quit") != NULL)
{
break;
}
}
shmdt(p);//解除映射 //int shmctl(int shmid, int cmd, struct shmid_ds *buf);
shmctl(shmid,IPC_RMID,NULL); //删除
semctl(semid,,IPC_RMID);//删除信号灯
return ;
}

测试:

例子2中通过两个2值信号灯实现同步操作,先把写数据到共享内存的信号灯初始化值为1,读共享内存的信号灯初始化为0,让写操作的进程先操作共享内存

linux 两个进程通过 共享内存 通信例子的更多相关文章

  1. Windows进程间共享内存通信实例

    Windows进程间共享内存通信实例 抄抄补补整出来 采用内存映射文件实现WIN32进程间的通讯:Windows中的内存映射文件的机制为我们高效地操作文件提供了一种途径,它允许我们在WIN32进程中保 ...

  2. linux使用共享内存通信的进程同步退出问题

    两个甚至多个进程使用共享内存(shm)通信,总遇到同步问题.这里的“同步问题”不是说进程读写同步问题,这个用信号量就好了.这里的同步问题说的是同步退出问题,到底谁先退出,怎么知道对方退出了.举个例子: ...

  3. C# .Net 多进程同步 通信 共享内存 内存映射文件 Memory Mapped 转 VC中进程与进程之间共享内存 .net环境下跨进程、高频率读写数据 使用C#开发Android应用之WebApp 分布式事务之消息补偿解决方案

    C# .Net 多进程同步 通信 共享内存 内存映射文件 Memory Mapped 转 节点通信存在两种模型:共享内存(Shared memory)和消息传递(Messages passing). ...

  4. PHP进程通信基础——信号量+共享内存通信

    PHP进程通信基础--信号量+共享内存通信 由于进程之间谁先执行并不确定,这取决于内核的进程调度算法,其中比较复杂.由此有可能多进程在相同的时间内同时访问共享内存,从而造成不可预料的错误.信号量这个名 ...

  5. C# 进程间共享内存通信方式

    从别处看到一篇文章做进程间通信很好使,唯一的问题是,需要注意using的用法,Using有个用法3, using 语句允许程序员指定使用资源的对象应当何时释放资源.using 语句中使用的对象必须实现 ...

  6. linux c编程:Posix共享内存区

    Posix共享内存区:共享内存是最快的可用IPC形式.它允许多个不相关(无亲缘关系)的进程去访问同一部分逻辑内存.如果需要在两个进程之间传输数据,共享内存将是一种效率极高的解决方案.一旦这样的内存区映 ...

  7. linux下的进程间通信之共享内存

    概念:这种机制允许两个或多个进程通过把公共数据结构放入一个共享内存区来访问它们.如果进程要访问这种数据结构所在的共享内存区,就必须在自己的地址空间中增加一个新线性区,新线性区映射与这个共享内存区相关的 ...

  8. linux下查询进程占用的内存方法总结

    linux下查询进程占用的内存方法总结,假设现在有一个「php-cgi」的进程 ,进程id为「25282」.现在想要查询该进程占用的内存大小.linux命令行下有很多的工具进行查看,现总结常见的几种方 ...

  9. Linux下php-fpm进程过多导致内存耗尽问题

    这篇文章主要介绍了解决Linux下php-fpm进程过多导致内存耗尽问题,需要的朋友可以参考下   最近,发现个人博客的Linux服务器,数据库服务经常挂掉,导致需要重启,才能正常访问,极其恶心,于是 ...

随机推荐

  1. 使用IntelliJ IDEA配置Maven(详细操作)

    一,下载Maven 进入官网http://maven.apache.org/  点击Download 找到如下图所示的区域,注意你的操作系统. 点击安装你所需要的安装包,下载,解压. 二,Maven环 ...

  2. SpringMVC学习(4):数据绑定1 @RequestParam

    在系列(3)中我们介绍了请求是如何映射到一个action上的,下一步当然是如何获取到请求中的数据,这就引出了本篇所要讲的内容-数据绑定. 首先看一下都有哪些绑定数据的注解: 1.@RequestPar ...

  3. Spring学习笔记(5)——IoC再度学习

    学习过Spring框架的人一定都会听过Spring的IoC(控制反转) .DI(依赖注入)这两个概念,对于初学Spring的人来说,总觉得IoC .DI这两个概念是模糊不清的,是很难理解的,今天和大家 ...

  4. bat批处理----set

    转载自    https://www.cnblogs.com/Braveliu/p/5081084.html [1]set命令简介 set,设置. [2]set命令使用 1. 打印系统环境变量.set ...

  5. MS14-068利用

    漏洞原理详情后续补上:kerberos 协议实现过程中的某些 bug,致使普通域用户可以任意伪造高权限 PAC,去请求 TGS 从而导致的权限提升,漏洞现在很少遇到了. 一.利用 需要拥有一个域账号的 ...

  6. mysql binlog相关

    1.清除的binlog   删除所有binlog日志,新日志编号从头开始 RESET MASTER;   删除mysql-bin.XXXX之前所有日志 PURGE MASTER LOGS TO 'my ...

  7. 解决虚拟机克隆的linux系统ip无法正常使用问题

    当我们克隆centos虚拟机无法正常获取IP地址,重启网卡也提示Bringing up interface eth0:  Device eth0 does not seem to be present ...

  8. 游戏game

    1.log4cxx 日志 2.protobuf  数据交互(类似json 3.boost.asio  网络库 4.boost.python 脚本支持 5.语法树 +  c++处理excel资源

  9. appium1.4+华为8.0执行自动化脚本,报启动session失败,原因是unicode_ime_apk\Uni codeIME-debug.apk在手机上已存在,再次安装失败,导致启动session失败,解决办法:换高版本的appium

    最开始做Android自动化测试时,通过执行脚本发现报,已安装UnicodeIME-debug.apk,再次安装失败,当时觉得这个apk对我来说没用,就把D:\Program Files (x86)\ ...

  10. ansible-继续普通用户权限运行

    ansible 远程以普通用户执行命令   1. ansible 10.0.0.1 -m raw -a "date" -u www 2.在ansible的主机配置文件中指定ssh_ ...