文章目录[隐藏]
- 文章目录
- 1.命名管道
- 2.共享内存
- 3.共享内存的接口
- 4.共享内存的标识符和共享内存的操作句柄的区别是什么?
- 5.删除共享内存
文章目录
- 1.命名管道
- 2.共享内存
- 3.共享内存的接口
- 4.共享内存的标识符和共享内存的操作句柄的区别是什么?
- 5.删除共享内存
1.命名管道
- 命名管道也是内核中开辟的一段缓存区,这段缓存区是有标识符的:意味着:不同的进程,不需要有亲缘关系,只需要通过标识符就能找到该缓存区
- 命名管道的创建
- mkfifo [命名管道文件]
p:代表的文件类型为管道文件 - 函数创建
mkfifo函数 - 特性
- 命名管道的生命周期也是跟随者进程的
- 命名管道是具有标识符的
- 其他的特性和匿名管道是一致的
2.共享内存
- 原理
- 在物理内存当中开辟了一段空间
- 这段物理内存的空间,可以被不同的进程附加到自己的共享区当中
- 附加的进程,通过操作共享区,来交换数据
3.共享内存的接口
- 创建共享内存
- int shmget(key_t key, size_t size, int shmflg)
key:共享内存标识符,这个标识符相当于共享内存的身份证,程序员第一次创建的时候,可以随意给值,只要和当前操作系统当中的其他的共享内存标识符不重复
size:共享内存的大小,单位:字节
shmflg:① IPC_CREAT:如果共享内存不存在,则创建共享内存
② IPC_EXCL:需要搭配IPC_CREAT使用,这样的宏在搭配使用的时候,还是采用按位或的方式(核心就是位图)
eg:IPC_CREAT | IPC_EXCL:
如果想要获取的共享内存,已经存在,则报错。如果想要获取的共享内存,是刚刚新创建出来的共享内存,则返回操作句柄 - 总结:使用shmget这个函数的时候一定要自己刚刚创建出来的共享内存
- 返回值:返回的是共享内存的操作句柄
- 共享内存命令
- ipcs:查看共享内存
- ipcrm -m [shmid]:删除共享内存
- 共享内存的声明周期是跟随着操作系统内核的
- 将共享内存附加到进程
- void shmat(int shmid, const void shmaddr, int shmflg)
shmid:共享内存的操作句柄
shmaddr:附加到共享内存的什么虚拟地址,允许传递NULL值,让操作系统帮我们选择附加到共享内存区当中的那个地址,这个地址通过该函数的返回值返回给我们
shmflg:① SHM_RDONLY:规定当前进程只能对共享内存进行读操作
② 0:可读可写 - 返回值:返回共享内存附加到共享区的地址
- 结论:进程在读取共享内存的时候,是访问,并不是拿走!
- 将共享内存和进程分离
- int shmdt(const void *shmaddr)
shmaddr:shmat的返回值 - 操作共享内存
- int shmctl(int shmid, int cmd, struct shmid_ds *buf)
shmid:共享内存操作句柄
cmd:告诉shmctl函数需要做什么操作
① IPC_STAT:获取当前共享内存的属性信息,放在buf中,buf是出参
② IPC_SET:设置共享内存的属性信息,是用buf来进行设置的,buf是入参
③ IPC_RMID:删除共享内存,buf可以直接传递NULL
buf:共享内存的结构体
4.共享内存的标识符和共享内存的操作句柄的区别是什么?
- 标识符:是用来标识共享内存的,相当于共享内存的身份证,意味着不同的进程可以通过标识符找到这个共享内存
- 操作句柄:进程可以通过操作句柄来对共享内存进行操作(,附加,分离,删除)
5.删除共享内存
- 当使用shmctl或者使用ipcrm,删除共享内存之后,共享内存就实际被释放掉了
- 当共享内存被释放掉之后,共享内存的标识符会被设置为0x00000000,表示其他进程不能通过之前的标识符找到该共享内存,并且共享内存的状态会被设置为dest(destroy)
- 当共享内存被释放掉之后,但是还是有进程在附加着共享内存,当前描述共享内存的结构体并没有被释放,直到当前共享内存的附加进程数量为0的时候才会释放掉。
© 版权声明