2015.3.4
星期三 阴天

进程间通信:IPC

文件对象:记录文件描述符,文件开关等

IPC标示符:系统全局的流水号
两个进程要通信,打开的是唯一的对象进行通讯,通过key操作

XSI IPC:消息队列,信号量,共享内存。

ipcs 查看ip对象共享内存,信号量,消息队列等信息
ipcrm 删除一个IP对象

Linux为用户提供了完善的,强大的网络功能
完善的内置网络:其他操作系统不包含如此紧密的和内核结合在一起的网络部分

共享内存标示符的获取有两种方法:ftok(pathname,id)另一个是KEY = IPC_CREATE

共享内存的创建,映射,读取状态等操作:(下面这个程序很短,但是基本包含了大部分共享内存函数的应用,程序入门非常好)

消息队列的实例程序;

#include <unistd.h>
#include <sys/ipc.h>
#include <sys/msg.h>

#define KEY 1234
#define TEXT_SIZE 48

struct msgbuffer
{
long mtype;
char mtext[TEXT_SIZE];
}msgp;

int mian(int argc, char **argv)
{
int msqid;
msqid = msgget(KEY, IPC_CTEAT | 0600); 建立消息队列

if(fork() == 0)
{
msgp.mtype = 1;
strcpy(msgp.mtext,"hi! I am child process!\n");
msgsnd(msqid,&msgp,TEXT_SIZE,0); 传送消息至队列
return ;
}

else
{
sleep(3); 等子进程执行完毕
msgrcv(msqid,&msgp,TEXT_SIZE,0,0); 接收队列的信息
printf("parent receive mtext : %s",msgp.mtext);
msgctl(msqid,IPC_RMID,NULL): 删除消息队列
}
}

#include <unistd.h>
#include <sys/ipc.h>

#include <sys/shm.h>

#define KEY 1234
#define SIZE 1024

int main(int argc, char **argv)
{
int shmid;
char *shmaddr;
struct shmid_ds buf;

shmid = shmget(KEY,SIZE,IPC_CREAT | 0600); 建立共享内存

if(fork() == 0) 子进程
{
shmaddr = (char *)shmat(shmid,NULL,0); 子进程映射共享内存
strcpy(shmaddr,"hi! I an child process!\n"); 对共享内存写入数据
shmdt(shmaddr); 释放子进程虚拟内存
return;
}

{ 父进程
sleep(3); 等待子进程执行完毕
shmctl(shmid,IPC_STAT,&buf); 取得共享内存的状态
printf("shm_segsz = %d bytes\n",shm_segsz);
printf("shm_cpid = %d bytes\n",shm_cpid);
printf("shm_lpid = %d bytes\n",shm_lpid);

shmaddr = (char *)shmat(shmid,NULL,0); 父进程映射共享内存
printf("%s",shmaddr); 显示共享内存内容
shmdt(shmaddr); 释放父进程虚拟内存
shmctl(shmid,IPC_RMID,NULL); 删除共享内存
}
}

共享内存的另一个程序:

#include <sys/shm.h>
#include <sys/stat.h>
#include <sys/ipc.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>

#define BUFSZ 2048

int main(int argc, char **argv)
{
int shmid;
char *shmaddr;

if((shmid = shmget(IPC_PRIVATE,BUFSZ,0666)) < 0); IPC_PRIVATE是指定键,保证了能创建一个新的IPC,缺点是获得的标识符写入
{ 了文件中,用户用的时候需要进入文件读取
perror("shmget");
exit(-1);
}

printf("###create shared - memory:%d\n",shmid);

system("ipcs -m | grep 666"); 相当于执行 ipcs -m

if((shmaddr = (char *)shmat(shmid,NULL,0))<(char *)0)
{
perror("shmat");
exit(-1);
}

printf("###create shared - memory\n");

system("ipcs -m | grep 666");

if((shmdt(shmaddr)) < 0)
{
perror("shmdt");
exit(-1);
}

printf("###create shared - memory\n");

system("ipcs -m | grep 666");

shmctl(shmid, IPC_RMID, NULL); 删除共享内存段
printf("###create shared - memory\n");

system("ipcs -m | grep 666");

return 0;

}

创建共享内存和信号灯并初始化,创建子进程,父进程读取标准输入并写入共享内存,子进程打印共享内存中的内容,双方通过信号灯进行同步;

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>

#include <sys/sem.h>

#define READ 0
#define WRITE 1

union semun 这个联合体需要自己定义
{
int val;
struct semid_ds *buf;
unsigned short *array;
struct seminfo *_buf;
};

int shmid; 定义的一些全局变量:
int semid;
pid_t pid;

int main()
{
key_t key;
char *shmaddr;
union semun myun;
struct sembuf buf;

if((key = ftok("./",'a')) == -1) 获得键值
{
perror("faile to ftok");
exit(-1);
}

if((shmid = shmget(key,64,IPC_CREAT|0666)) < 0) 建立共享内存,如果存在就取内存ID
{
perror("fail to shmget");
exit(-1);
}

shmaddr = (char*)shmat(shmid,NULL,0); 映射虚拟内存,下面调用fork(),会将映射继承到子进程中,这是创建进程间相同映射的一种方式

if((semid = semget(key,2,IPC_CREAT|0666)) < 0) 创建信号集,信号集中有两个信号
{
perror("fail to semget");
exit(-1);
}

myun.val = 0; 这是一个初始化操作,表示信号量的个数
semctl(semid,READ,SETVAL,myun);
myun.val = 1;
semctl(semid,WRITE,SETVAL,myun);

if((pid = fork()) < 0) 创建子进程
{
perror("fail to fork");
exit(-1);
}
else if(pid == 0)
{
buf.sem_num = READ; 信号个数0
buf.sem_op = -1; p操作,信号量小于等于0则阻塞 (sem_wait()函数) 3.在2之后开始执行
buf.sem_flg = 0;
semop(semid,&buf,1);
printf("read from shm : %s",shmaddr);
buf.sem_num = WRITE; 信号个数1
buf.sem_op = 1; v操作 (sem_post()函数)将信号 4.将信号量个数置0,子进程等待。
buf.sem_flg = 0;
semop(semid,&buf,1);
}
else
{
buf.sem_num = WRITE; 信号个数
buf.sem_op = -1; p操作 (sem_post()函数) 1.执行
buf.sem_flg = 0;
semop(semid,&buf,1);
printf("write to shm : ");
fgets(shmaddr,64,stdin);
buf.sem_num = READ; 信号个数0
buf.sem_op = 1; V操作 (sem_wait()函数) 2.唤醒子进程的等待程序
buf.sem_flg = 0;
semop(semid,&buf,1);
}

return 0;
}

***************************************************************************************************************************************************************
***************************************************************************************************************************************************************
***************************************************************************************************************************************************************
***************************************************************************************************************************************************************

进程间通信IPC:消息队列,信号量,共享内存的更多相关文章

  1. Linux进程间通信(消息队列/信号量+共享内存)

    写在前面 不得不说,Deadline果真是第一生产力.不过做出来的东西真的是不堪入目,于是又花了一早上重写代码. 实验内容 进程通信的邮箱方式由操作系统提供形如 send()和 receive()的系 ...

  2. 进程间通信之信号量、消息队列、共享内存(system v的shm和mmap)+信号signal

    进程间通信方式有:System v unix提供3种进程间通信IPC:信号量.消息队列.共享内存.此外,传统方法:信号.管道.socket套接字. [注意上述6种方式只能用户层进程间通信.内核内部有类 ...

  3. Linux进程间通信:管道,信号量,消息队列,信号,共享内存,套接字

    Linux下的进程通信手段基本上是从UNIX平台上的进程通信手段继承而来的.而对UNIX发展做出重大贡献的两大主力AT&T的贝尔实验室及BSD(加州大学伯克利分校的伯克利软件发布中心)在进程间 ...

  4. boost进程间通信经常使用开发一篇全(消息队列,共享内存,信号)

    本文概要: 敏捷开发大家想必知道并且评价甚高,缩短开发周期,提高开发质量.将大project独立为不同的小app开发,整个开发过程,程序可用可測,所以提高了总体的质量.基于这样的开发模式和开发理念,进 ...

  5. 8.7 进程间的通讯:管道、消息队列、共享内存、信号量、信号、Socket

    进程间的通讯 进程间为什么需要通讯? 共享数据.数据传输.消息通知.进程控制 进程间的通讯有哪些类型? 首先,联系前面讲过的知识,进程之间的用户地址空间是相互独立的,不能进行互相访问,但是,内核空间却 ...

  6. linux IPC 消息队列

    消息队列函数原型 在建立IPC通讯时(如消息队列,共享内存)必须建立一个ID值.通常情况下,这个ID值由ftok函数得到 #inlcude <sys/types.h> #include & ...

  7. IPC——消息队列

    Linux进程间通信——使用消息队列 下面来说说如何用不用消息队列来进行进程间的通信,消息队列与命名管道有很多相似之处.有关命名管道的更多内容可以参阅我的另一篇文章:Linux进程间通信——使用命名管 ...

  8. c/c++ linux 进程间通信系列4,使用共享内存

    linux 进程间通信系列4,使用共享内存 1,创建共享内存,用到的函数shmget, shmat, shmdt 函数名 功能描述 shmget 创建共享内存,返回pic key shmat 第一次创 ...

  9. [原创]chromium源码阅读-进程间通信IPC.消息的接收与应答

    chromium源码阅读-进程间通信IPC.消息的接收与应答   chromium源码阅读-进程间通信IPC.消息的接收与应答 介绍 chromium进程间通信在win32下是通过命名管道的方式实现的 ...

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

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

随机推荐

  1. 【Tarjan】+【SPFA】APIO2009 Atm

    一.算法介绍 tarjan——求解有向图强连通分量.这个算法在本人的一篇blog中有介绍,这里就不赘述了.贴上介绍tarjan的的blog链接:http://www.cnblogs.com/Maki- ...

  2. 【RabbitMQ】RabbitMQ在Windows的安装和简单的使用

    版本说明 使用当前版本:3.5.4 安装与启动 在官网上下载其Server二进制安装包,在Windows上的安装时简单的,与一般软件没什么区别. 安装前会提示你,还需要安装Erlang,并打开下载页面 ...

  3. QT笔记之不规则窗口的实现

    QT实现的不规则窗口,是根据图片的形状显示 1.去标题栏 2.设置窗口背景为透明色 3.最后给窗口设置背景色 注:背景图为镂空的 格式为.png 图片资源下载:http://pan.baidu.com ...

  4. FIFO页面置换算法

    本文以序列长度20的{ 7,0,1,2,0,3,0,4,2,3,0,3,2,1,2,0,1,7,0,1};以及页面4:为例: #include <stdio.h> #define Init ...

  5. 使用R进行倾向得分匹配

    pacman::p_load(knitr, wakefield, MatchIt, tableone, captioner)set.seed(1234)library(wakefield)df.pat ...

  6. 将php网站移到CentOS 6.7上[一]:yum安装lamp环境

    最近应老师要求,将一个网站从51php上转移到学校提供的服务器上,之前对Linux没有了解,一切都在百度百度百度.于是发现很多步骤自己做过后就忘了,现将有效步骤记录下来,以供下次参考. 原51php上 ...

  7. (原创)Windows和Linux间共享文件

    方法一: Windows创建目录sharedir,修改共享属性,对everyone开放读写权限. Linux下运行命令:# mount -t cifs //192.168.8.55/sharedir ...

  8. java高薪之路__005_IO流

    参考地址: 1. http://blog.csdn.net/yczz/article/details/38761237 File类 ObjectInputStream && Objec ...

  9. Android 连接webservice(利用谷歌提供的jar包)

    Android开发,需要连接webservice,之前就想用谷歌提供的jar包,下载地址:http://pan.baidu.com/s/1hqMTUHe 把它下载下来粘贴到libs文件夹下即可: 网上 ...

  10. Nginx 1.10.2 php 7 环境安装

    1.安装编译工具和库文件,红色部分提示在centos镜像站点上查不到包,用yum安装的时候要认真看那些包没有找到,用yum的时候尽量不要使用-y选项 yum install gcc automake ...