IPC---共享内存
#include <sys/ipc.h>
#include <sys/shm.h> int shmget(key_t key, size_t size, int shmflg);
#include <sys/types.h>
#include <sys/shm.h> void *shmat(int shmid, const void *shmaddr, int shmflg);
第一次创建完共享内存时,它还不能被任何进程访问,shmat函数的作用就是用来启动对该共享内存的访问,并把共享内存连接到当前进程的地址空间。
调用成功时返回一个指向共享内存第一个字节的指针,如果调用失败返回-1.
#include <sys/types.h>
#include <sys/shm.h>
int shmdt(const void *shmaddr);
该函数用于将共享内存从当前进程中分离。注意:将共享内存分离并不是删除它,只是使该共享内存对当前进程不再可用。
#include <sys/ipc.h>
#include <sys/shm.h> int shmctl(int shmid, int cmd, struct shmid_ds *buf);
与信号量的semctl函数一样,用来控制共享内存
struct shmid_ds {
struct ipc_perm shm_perm; /* Ownership and permissions */
size_t shm_segsz; /* Size of segment (bytes) */
time_t shm_atime; /* Last attach time */
time_t shm_dtime; /* Last detach time */
time_t shm_ctime; /* Last change time */
pid_t shm_cpid; /* PID of creator */
pid_t shm_lpid; /* PID of last shmat(2)/shmdt(2) */
shmatt_t shm_nattch; /* No. of current attaches */
...
};
示例
memory_write.c为创建共享内存并向里面写入由中断输入得数据,memory_read.c为读取共享内存中的内容。两个程序运行在两个不相关的进程中,为了保证两个进程间的读写同步,设置了WR_RD标记。
memory_write.c的程序如下
#include <stdio.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdlib.h>
#include <string.h> #define MAX_SIZE 2048
#define BUF_SIZE 2047 #define ERR_EXIT(m) \
do \
{ \
perror(m); \
exit(EXIT_FAILURE); \
}while(0) struct share_memory
{
int WR_RD; //非0表示可写,0表示可读
char text[MAX_SIZE];
}; int main()
{
int shmid;
char buffer[BUF_SIZE+1]; //用于保存输入的文本
shmid = shmget(1234,sizeof(struct share_memory),0666 | IPC_CREAT); //创建共享内存
if(shmid==-1)
ERR_EXIT("shmget failed\n"); struct share_memory *sh_mem;
void *sh=NULL;
sh = shmat(shmid,0,0); //将共享内存连接到当前进程的地址空间
if(sh ==(void *)-1)
ERR_EXIT("shmat failed\n");
else
printf("Memory attached at9 %x\n",(int)sh); sh_mem=(struct share_memory *)sh;
sh_mem->WR_RD=1; // 置为可写
int flag=1; while(1)
{
while(!sh_mem->WR_RD) //共享内存为可读,表示内存里的数据还未被读取,不能写,此时堵塞
{
if(flag)
{
printf("memory is not empty,waiting.....\n");
flag=0;
} sleep(1);
} flag=1;
/*向共享内存写入数据*/
printf("Enter some text(Enter 'end' to quit): \n");
fgets(buffer,BUF_SIZE,stdin);
strncpy(sh_mem->text,buffer,sizeof(buffer)); sh_mem->WR_RD=0; //写完数据之后设置共享内存可读 if(strncmp(buffer,"end",3)==0)
break; //输入end结束循环 } if(shmdt(sh)==-1)
ERR_EXIT("shmdt failed\n"); //将共享内存从当前进程分离 sleep(2);
return 0; }
memory_read.c的程序如下
#include <stdio.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdlib.h>
#include <sys/types.h>
#include <unistd.h> #define MAX_SIZE 2048 #define ERR_EXIT(m) \
do \
{ \
perror(m); \
exit(EXIT_FAILURE); \
}while(0) struct share_memory
{
int WR_RD; //非0表示可写,0表示可读
char text[MAX_SIZE];
}; int main()
{
int shmid; srand(getpid()); shmid = shmget(1234,sizeof(struct share_memory),0666 | IPC_CREAT); //创建共享内存
if(shmid==-1)
ERR_EXIT("shmget failed\n"); struct share_memory *sh_mem;
void *sh=NULL;
sh = shmat(shmid,0,0); //将共享内存连接到当前进程的地址空间
if(sh ==(void *)-1)
ERR_EXIT("shmat failed\n");
else
printf("Memory attached at %x\n",(int)sh); sh_mem=(struct share_memory*)sh;
//sh_mem->WR_RD=ture; // 置为可写 while(1) if(!sh_mem->WR_RD) //共享内存为可读,从共享内存读取数据
{
printf("your wrote: %s", sh_mem->text);
sleep(rand()%3);
sh_mem->WR_RD=1; //读完数据之后设置共享内存可写 if(strncmp(sh_mem->text,"end",3)==0)
break; //输入end结束循环
}
else //表示内存为空,不能读,此时堵塞
{
//printf("memory is empty,waiting.....\n");
sleep(1);
} if(shmdt(sh)==-1)
ERR_EXIT("shmdt failed\n"); //将共享内存从当前进程分离 if(shmctl(shmid,IPC_RMID,0)==-1)
ERR_EXIT("shmctl failed\n"); return 0; }
IPC---共享内存的更多相关文章
- Linux IPC 共享内存
共享内存 共享内存(shared memory)是最简单的Linux进程间通信方式之一. 使用共享内存,不同进程可以对同一块内存进行读写. 由于所有进程对共享内存的访问就和访问自己的内存空间一样,而不 ...
- linux IPC共享内存
共享内存相关函数 获得一个共享存储标识符 #include <sys/ipc.h> #include <sys/shm.h int shmget(key_t key, size_t ...
- IPC——共享内存
Linux进程间通信——使用共享内存 下面将讲解进程间通信的另一种方式,使用共享内存. 一.什么是共享内存 顾名思义,共享内存就是允许两个不相关的进程访问同一个逻辑内存.共享内存是在两个正在运行的 ...
- Linux环境进程间通信(五): 共享内存(下)
linux下进程间通信的几种主要手段: 管道(Pipe)及有名管道(named pipe):管道可用于具有亲缘关系进程间的通信,有名管道克服了管道没有名字的限制,因此,除具有管道所具有的功能外,它还允 ...
- 【转载】ipcs与Linux共享内存
一.共享内存相关知识 所谓共享内存,就是多个进程间共同地使用同一段物理内存空间,它是通过将同一段物理内存映射到不同进程的 虚拟空间来实现的.由于映射到不同进程的虚拟空间中,不同进程可以直接使用,不需要 ...
- 进程间通信之信号量、消息队列、共享内存(system v的shm和mmap)+信号signal
进程间通信方式有:System v unix提供3种进程间通信IPC:信号量.消息队列.共享内存.此外,传统方法:信号.管道.socket套接字. [注意上述6种方式只能用户层进程间通信.内核内部有类 ...
- <转>Linux环境进程间通信(五): 共享内存(下)
http://www.ibm.com/developerworks/cn/linux/l-ipc/part5/index2.html 系统调用mmap()通过映射一个普通文件实现共享内存.系统V则是通 ...
- Linux环境进程间通信: 共享内存
Linux环境进程间通信: 共享内存 第一部分 共享内存可以说是最有用的进程间通信方式,也是最快的IPC形式.两个不同进程A.B共享内存的意思是,同一块物理内存被映射到进程A.B各自的进程地址空间.进 ...
- (转)Linux环境进程间通信系列(五):共享内存
原文地址:http://www.cppblog.com/mydriverc/articles/29741.html 共享内存可以说是最有用的进程间通信方式,也是最快的 IPC 形式.两个不同进程 A ...
- Linux进程间通信(四) - 共享内存
共享内存的优势 采用共享内存通信的一个显而易见的好处是效率高,因为进程可以直接读写内存,而不需要任何数据的拷贝.对于像管道和消息队列等通信方式,则需要在内核和用户空间进行四次的数据拷贝,而共享内存则只 ...
随机推荐
- Struts2 action的单例与多例
struts 2的Action是多实例的并非单例,也就是每次请求产生一个Action的对象.原因是:struts 2的Action中包含数据,例如你在页面填写的数据就会包含在Action的成员变量里面 ...
- 用libsvm进行回归预测
最近因工作需要,学习了台湾大学林智仁(Lin Chih-Jen)教授等人开发的SVM算法开源算法包. 为了以后方便查阅,特把环境配置及参数设置等方面的信息记录下来. 林教授年轻时照片 SVM属于十大挖 ...
- vim 打开Linux下文件每一行后面都有^M的样式
由于服务器不是我一个人在操作,在修改apache配置文件时发现了一个很奇怪的问题,vim编辑打开配置文件发现后面都有一个^M的标记 虽然不会影响服务的运行,但总感觉不对劲,所以在此我尝试用替换的方式来 ...
- 来聊聊apply和call
今天在群里讨论的时候,看到有人问apply怎么使用,突然想起自己刚接触这个方法的时候,也是一样的摸不着头脑. 记得当时上网看了很多讲解,可实际用的时候还是感觉有些蒙蒙哒orz.... 后来想一想,也许 ...
- AngularJS API之equal比较对象
使用情况 1 首先,所有满足 a === 3 这种的对象,在angular.equals(a,b)中都会返回真 2 所有对象的类型,以及属性值都相同的,也会返回真 3 NaN和NaN也会返回真(在ja ...
- go语言mongdb管道使用
原始json: { "listsn": "", "code": "fwq_add", "detail" ...
- POJ 3744 Scout YYF I
分段的概率DP+矩阵快速幂 Scout YYF I Time Limit: 1000MS Memory Limit: 65536K Total Sub ...
- 天翼宽带政企网关B2-1P 如何获得超级管理员账号?
RT 用useradmin没办法做NAT,想进telecomadmin里面看看,,,,,并且已经使用过nE7jA%5m这个密码登录,没有用! 求办法!!! 最佳答案 查找超级管理员密码方法: 1.用光 ...
- Mac Pro 编译安装 Nginx 1.8.1
#下载相关源码包,统一放到 /usr/local/src 目录下: http://nginx.org/download/nginx-1.8.1.tar.gz http://zlib.net/zlib- ...
- java之google style
Google的Java编码规范英文版: http://google-styleguide.googlecode.com/svn/trunk/javaguide.html Google的Java编码规范 ...