共享内存通信
1.概述
共享内存,顾名思义就是允许两个不相关的进程访问同一个逻辑内存,共享内存是两个正在运行的进程 之间共享和传递数据的一种非常有效的方式。不同进程之间共享的内存通常为同一段物理内存。进程可以将同一段物理内存连接到他们自己的地址空间中,所有的进程都可以访问共享内存中的地址。如果某个进程向共享内存写入数据,所做的改动将立即影响到可以访问同一段共享内存的任何其他进程。
命令
(1)查看系统中的共享存储段
ipcs -m
(2)删除系统中的共享存储段
ipcrm -m [shmid]
函数
#include <sys/types.h>
#include <sys/shm.h>
(1) shmget()创建共享内存
int shmget(key_t key, size_t size, int shmflg);
key:由ftok生成的key标识,标识系统的唯一IPC资源
size:需要申请共享内存的大小
shmflg主要和一些标志有关。其中有效的包括IPC_CREAT和IPC_EXCL,
IPC_CREAT 如果共享内存不存在,则创建一个共享内存,否则打开操作。
IPC_EXCL 只有在共享内存不存在的时候,新的共享内存才建立,否则就产生错误。
(2) shmat()挂接共享内存
void *shmat(int shmid, const void *shmaddr, int shmflg);
shmid: 是shmget 返回的标识符
shmaddr: NULL,系统将自动选择一个合适的地址
shmflg; 若指定SHM_RDONLY位,则以只读方式连接此段,0是以读写方式连接此段
(3) shmdt()去关联共享内存
int shmdt(const void *shmaddr);
shmadr:连接以后返回的地址
(4)shmctl()销毁共享内存
int shmctl(int shmid, int cmd, struct shmid_ds *buf);
shmid: 共享存储段标识符
cmd:指定的执行操作,设置为IPC_RMID 时表示可以删除共享内存
buf: 设置为NULL即可
进程组和会话
概念和特性
进程组,每个进程组有一个领头进程。进程组是一个或多个进程的集合,通常它们与一组作业相关联, 可以接受来自同一终端的各种信号。
创建会话
创建一个会话需要注意以下6点注意事项:
1.调用进程不能是进程组组长,该进程变成新会话首进程(session header)
2.该进程成为一个新进程组的组长进程。
3.需有root权限(ubuntu不需要)
4.新会话丟弃原有的控制終端,该会话没有控制終端。
5.该调用进程是组长进程,则出错返回。
6.建立新会话时,先调用fork, 父进程终止,子进程调用setsidu
函数
#include <unistd.h>
getsid函数
pid_t getsid(pid_t pid)
1.pid为0表示察看当前进程session ID
2.ps ajx命令查看系统中的进程。参数a表示不仅列当前用户的进程,也列出所有其他用户的进程,参数x表示不仅列有控制终端的进程,也列出所有无控制终端的进程,参数j表示列出与作业控制相关的信息。
3.组长进程不能成为新会话首进程,新会话首进程必定会成为组长进程。
setsid函数
创建一个会话,并以自己的ID设置进程组ID,同时也是新会话的ID。
pid_t setsid(void);
成功:返回调用进程的会话ID;失败:-1,设置errno。
调用了setsid函数的进程,既是新的会长,也是新的组长。
系统守护进程介绍和创建
守护进程(daemon)是一类在后台运行的特殊进程,用于执行特定的系统任务。很多守护进程在系统 引导的时候启动,并且一直运行到系统关闭。另一些只在需要的时候才启动,完成任务后就自动结束。
创建守护进程步骤
1.创建子进程, 父进程退出 所有工作在子进程中进行形式上脱离了控制終端
2.在子进程中创建新会话 setsid)函数 使子进程完全独立出来,脱离控制
3.改变当前目录(比如为根目录) chdir()函数 防止占用可卸载的文件系统 也可以换成其它路径
4.重设文件权限掩码 umask()函数 防止继承的文件创建屏蔽宇拒绝某些权限 增加守护进程灵活性
5.关闭文件描述符。 继承的打开文件不会用到,浪费系统资源,无法卸载
6.开始执行守护进程核心工作守护进程退出处理程序模型