好久没更新博客了,最近几个月一直在忙项目,现在终于有时间进一步学习了。这次记录的是unix环境中共享内存的使用方法。

  

 在我理解,共享内存就是在内存中开辟一段空间,各个毫不相干的进程就可以通过访问这段内存中的资源,从而达到进程间通信的效果。由于是对同一块内存进行操作,没有涉及到资源的复制等操作,所以在IPC对象中共享内存速度和效率都是很高的。
     使用共享内存过程中需要的函数
     1、shmget
     该函数通过键值向内核申请一块共享内存
     函数原型:int  shmget(key_t key, size_t size, int shmflg);
     其中,key是每一个IPC结构都需要的键值;size是想要获取共享内存的大小,单位是字节;shmflag是获取共享内存的标志,一般为IPC_CREAT,当该key值对应的共享内存不存在时创建之,存在的话就打开。当创建共享内存时,还应添加创建共享内存的权限,如IPC_CREAT | 0666。该函数成功返回时得到共享内存的标示符,失败返回-1。
     2、shmat
     该函数将已申请到的共享内存连接到用户进程空间中
     函数原型:void * shmat(int shmid, const void *shmaddr, int shmflg);
     其中,shmid是共享内存的标识符;shmaddr指明将该内存映射到进程空间哪一块儿区域,推荐将其置NULL,由linux内核自动分配;flag为连接到进程空间时的权限选项,一般填0。该函数成功返回时返回该共享内存的首地址,失败返回-1。
     3、shmdt
     该函数将连接到用户进程的共享内存与进程分离
     函数原型:int shmdt(const void *shmaddr);
     其中,shmaddr为要分离的共享内存的首地址。成功返回0,失败返回-1。
     4、shmctl
     该函数用于控制得到的共享内存
     函数原型:int shmctl(int shmid, int cmd, struct shmid_ds *buf);
     内存中每一段共享内存都对应一个shmid_ds结构体用来维护。其中,cmd是要对shmid指向的共享内存执行的操作。当cmd为IPC_STAT时,取出shmid指向的共享内存对应的shmid_ds结构,并存储到buf中。当cmd为IPC_RMID时,删除shmid指向的共享内存段,此时buf填0。
 
  

======================分割线================================
 
     下面写两个程序来演示共享内存的使用方法。设计思路:write进程打开共享内存,并向其中循环写入数据,每次写完后等待read进程将其读走,再进行下一次写入。当write进程写入end时,两个进程都终止。贴代码:

/*write.c*/
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h> /*共享内存中缓冲区的大小*/
#define BUFFER_SIZE 2048 struct temp
{
int write_flag;
char share_buffer[BUFFER_SIZE]
}; int main(int argc, char const *argv[])
{
int shmid; /*循环开关*/
int running = ; char buffer[BUFFER_SIZE]; struct temp *share_temp; /*合成键值*/
key_t key = ftok(".", ); /*打开共享内存*/
if((shmid = shmget(key, sizeof(struct temp), IPC_CREAT)) == -)
{
printf("memory get failed\n");
exit(EXIT_FAILURE);
} printf("shmid is %d\n", shmid); /*映射共享内存*/
if((share_temp = shmat(shmid, NULL, )) == (void *)-)
{
printf("memory at failed\n");
exit(EXIT_FAILURE);
} /*循环写入数据*/
while(running)
{
while(share_temp->write_flag)
{
sleep();
printf("wait for read process\n");
}
fgets(buffer, BUFFER_SIZE, stdin);
strncpy(share_temp->share_buffer, buffer, BUFFER_SIZE); share_temp->write_flag = ;
if(strncmp(share_temp->share_buffer, "end", ) == )
running = ;
} /*分离共享内存*/
if((shmdt((void *)share_temp)) == -)
{
printf("memory fenli failed\n");
exit(EXIT_FAILURE);
} return ;
}
/*read.c*/
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h> /*共享内存中缓冲区的大小*/
#define BUFFER_SIZE 2048 struct temp
{
int write_flag;
char share_buffer[BUFFER_SIZE]
}; int main(int argc, char const *argv[])
{
int shmid; /*循环开关*/
int running = ; struct temp *share_temp; key_t key = ftok(".", ); /*创建共享内存*/
if((shmid = shmget(key, sizeof(struct temp), IPC_CREAT|)) == -)
{
printf("memory get failed\n");
exit(EXIT_FAILURE);
} printf("shmid is %d\n", shmid); /*映射共享内存*/
if((share_temp = shmat(shmid, NULL, )) == (void *)-)
{
printf("memory at failed\n");
exit(EXIT_FAILURE);
} /*循环读取数据*/
while(running)
{
if(share_temp->write_flag)
{
printf("write is %s\n", share_temp->share_buffer);
share_temp->write_flag = ;
} if(strncmp(share_temp->share_buffer, "end", ) == )
running = ;
} /*分离共享内存*/
if((shmdt((void *)share_temp)) == -)
{
printf("memory fenli failed\n");
exit(EXIT_FAILURE);
} /*删除共享内存*/
if((shmctl(shmid, IPC_RMID, )) == -)
{
printf("memory delete failed\n");
exit(EXIT_FAILURE);
} return ;
}

  下面说说写程序过程中遇到的问题:开始用普通用户执行read程序,创建共享内存的时候没有加上0666选项,结构返回错误信息memory at failed,即连接到进程空间失败,切换到root用户再执行就可以。一开始还以为linux歧视普通用户,不让普通用户使用共享内存。后来发现创建的时候加上权限0666就可以了,其他IPC对象也同理。

  如果有疑问或错误,欢迎指出。

UNIX环境下的共享内存的更多相关文章

  1. [转]Windows环境下利用“共享内存”实现进程间通信的C/C++代码---利用CreateFileMapping和MapViewOfFile

    http://blog.csdn.net/stpeace/article/details/39534361 进程间的通信方式有很多种, 上次我们说了最傻瓜的“共享外存/文件”的方法. 那么, 在本文中 ...

  2. Linux环境进程间通信(五): 共享内存(下)

    linux下进程间通信的几种主要手段: 管道(Pipe)及有名管道(named pipe):管道可用于具有亲缘关系进程间的通信,有名管道克服了管道没有名字的限制,因此,除具有管道所具有的功能外,它还允 ...

  3. Unix IPC之基于共享内存的计数器

    目的 本文主要实现一个基于共享内存的计数器,通过父子进程对其访问. 本文程序需基于<<Unix网络编程-卷2>>的环境才能运行.程序中大写开头的函数为其小写同名函数的包裹函数, ...

  4. Linux环境进程间通信(五): 共享内存(上)

    linux下进程间通信的几种主要手段: 管道(Pipe)及有名管道(named pipe):管道可用于具有亲缘关系进程间的通信,有名管道克服了管道没有名字的限制,因此,除具有管道所具有的功能外,它还允 ...

  5. Tomcat集群环境下session共享方案 通过memcached 方法实现

    对于web应用集群的技术实现而言,最大的难点就是:如何能在集群中的多个节点之间保持数据的一致性,会话(Session)信息是这些数据中最重要的一块.要实现这一点, 大体上有两种方式:一种是把所有Ses ...

  6. 缓存利器、Lua模块下的共享内存

    上一节讲到了worker进程的共享内存,它利用丰富的指令使数据的缓存操作变得非常简单,但它也存在一些缺点. 1.worker进程之间会有锁竞争,在高并发的情况下会增加性能开销.2.只支持Lua布尔值. ...

  7. win32下进程间通信——共享内存

    一.引言     在Windows程序中,各个进程之间常常需要交换数据,进行数据通讯.WIN32 API提供了许多函数使我们能够方便高效的进行进程间的通讯,通过这些函数我们可以控制不同进程间的数据交换 ...

  8. Linux环境编程之共享内存区(一):共享内存区简单介绍

    共享内存区是可用IPC形式中最快的.一旦内存区映射到共享它的进程的地址空间,进程间数据的传递就不再涉及内核.然而往该共享内存区存放信息或从中取走信息的进程间通常须要某种形式的同步.不再涉及内核是指:进 ...

  9. 【SpringBoot】spring-session-data-redis 解决集群环境下session共享

    为什么会产生Session共享问题   集群情况下,session保存在各自的服务器的tomcat中,当分发地址至不同服务时,导致sesson取不到,就会产生session共享问题. 解决方案 负载均 ...

随机推荐

  1. Java-Android 之电话拨号源码

    file:///F:/workspace3/Android_ver2.4/src/cn/szy/com/MainActivity.java package cn.szy.com; import and ...

  2. 单例模式,多种实现方式JAVA

    转载请注明出处:http://cantellow.iteye.com/blog/838473 第一种(懒汉,线程不安全): public class Singleton { private stati ...

  3. JQ 日期格式化

    将字符转换为日期格式: function getDate(strDate) { var date = eval('new Date(' + strDate.replace(/\d+(?=-[^-]+$ ...

  4. WPF Binding值转换器ValueConverter使用简介(一)

    WPF.Silverlight及Windows Phone程序开发中往往需要将绑定的数据进行特定转换,比如DateTime类型的时间转换为yyyyMMdd的日期,再如有一个值是根据另外多组值的不同而异 ...

  5. java 生成pdf报表

    public void saveMapAddressInfo(String orderCode){ try{ List<Leads> leadses = leadsService.find ...

  6. maven常见命令总结

    1.maven vm配置 -Xms512m -Xmx1024m -XX:PermSize=256m 2.启动tomcat  clean package -Dpackage.environment=de ...

  7. Oracle dblink 使用详解

    1.dblink简介 dblink(Database Link)数据库链接就是数据库的链接,跨本地数据库 2.使用语法详解 基本语法 CREATE [SHARED][PUBLIC] database ...

  8. Visual Studio 2013环境下操作vc6/vc7/vc8等低版本平台项目【编译|生成|调试】

    现代化的开发环境,微软一直在推出更新换代,我们所处的技术环境在日新月异的变化:不过在中国多数人们一边疲惫的追赶着时代的步伐,一边坚守着自己所获悉所掌握的那些紧吧吧的知本.对技术工具的掌握并非他们所想要 ...

  9. 封装JDBC:实现简单ORM框架lfdb

    作者:Vinkn 来自http://www.cnblogs.com/Vinkn/ 一.简介 框架就是一组可重用的构件,LZ自己写的姑且就叫微型小框架:lfdb.LZ也对其他的ORM框架没有什么了解,现 ...

  10. 支付宝支付错误 系统忙:错误代码AE150002999

    今天出现支付宝支付错误,支付系统繁忙请稍等,错误码AE150002999.测试了其他连个收款方,可跳转到正常扫码支付页面,排除了代码错误. 在登录支付宝商家中,也签约了“即时到帐”功能.度娘和查看都找 ...