目录
共享内存
1. 特点
1)共享内存是一种最为高效的进程间通信方式,进程可以直接读写内存,而不需要任何数据的拷贝
2)为了在多个进程间交换信息,内核专门留出了一块内存区,可以由需要访问的进程将其映射到自己的私有地址空间
3)进程就可以直接读写这一内存区而不需要进行数据的拷贝,从而大大提高的效率。
4)由于多个进程共享一段内存,因此也需要依靠某种同步机制,如互斥锁和信号量等
2. 步骤
1)创建key值(ftok)
2)创建或打开共享内存
3)映射共享内存到用户空间
4)撤销映射
5)删除共享内存
3. 函数
1) 创建key值 ftok
key_t ftok(constchar* pathname,int proj_id);
功能:
产生一个独一无二的key值
参数:
Pathname:已经存在的可访问文件的名字
Proj_id:一个字符(因为只用低8位)
返回值:
成功:key值
失败:-1
int main(int argc, char const *argv[])
{
key_t key = ftok("./dest.c", 'a');
if (key < 0)
{
perror("ftok err");
return -1;
}
printf("%#x\n", key);
}
key值组成:
- 前两位是字符的ASCII值
- 中间两位是系统编号
- 后四位是文件对应inode号的后四位。
2) 创建或打开共享内存shmget
int shmget(key_t key, size_t size,int shmflg);
功能:
创建或打开共享内存
参数:
key: 键值
size: 共享内存的大小
shmflg: IPC_CREAT|IPC_EXCL(判错)|0666
返回值:
成功 shmid
出错 -1
//创建或打开共享内存
int shmid = shmget(key, 128, IPC_CREAT | IPC_EXCL | 0666);
if (shmid <= 0)
{
if (errno == EEXIST)
{
shmid = shmget(key, 128, 0666);
}
else
{
perror("shmget err");
return -1;
}
}
printf("shmid:%d\n", shmid);
3) 映射共享内存到用户空间shmat
void* shmat(int shmid,constvoid* shmaddr,int shmflg);
功能:
映射共享内存,即把指定的共享内存映射到进程的地址空间用于访问
参数:
shmid 共享内存的id号
shmaddr 一般为NULL,表示由系统自动完成映射
如果不为NULL,那么由用户指定
shmflg:SHM_RDONLY就是对该共享内存只进行读操作
0 可读可写
返回值:
成功:完成映射后的地址,
失败:-1的地址
char *p = shmat(shmid, NULL, 0);
if (p == (void *)-1)
{
perror("shmat err");
return -1;
}
printf("hello\n");
//读写操作
//向共享内存里写
fgets(p,32,stdin);
printf("%s\n",p);
4) 取消映射shmdt
int shmdt(constvoid* shmaddr);
功能:
取消映射
参数:
要取消的地址
返回值:
成功0
失败的-1
5) 删除共享内存shmctl
int shmctl(int shmid,int cmd,structshmid_ds* buf);
功能:
(删除共享内存),对共享内存进行各种操作
参数:
shmid :共享内存的id号
cmd:
IPC_STAT 获得shmid属性信息,存放在第三参数
IPC_SET 设置shmid属性信息,要设置的属性放在第三参数
IPC_RMID删除共享内存,此时第三个参数为NULL即可
用法:shmctl(shmid,IPC_RMID,NULL);
shmdt(p);//取消映射
shmctl(shmid,IPC_RMID,NULL);//删除共享内存
4.命令
ipcs -m :查看系统中共享内存
ipcrm -m shmid:删除shmid号的共享内存
消息队列
1. 特点
消息队列是IPC对象的一种
消息队列由消息队列ID来唯一标识
消息队列就是一个消息的列表。用户可以在消息队列中添加消息、读取消息等。
消息队列可以按照类型来发送(添加)/接收(读取)消息
2. 步骤
1)创建key值(ftok)
2)创建或者打开消息队列
3)添加消息:按照类型将消息添加到已经代开的消息队列末尾
4)读取消息:可以按照类型读取消息
5)删除消息队列
3. 函数
1)创建或打开一个消息队列msgget
int msgget(key_t key,int flag);
功能:
创建或打开一个消息队列
参数:
key值
flag:创建消息队列的权限IPC_CREAT|IPC_EXCL|0666
返回值:
成功:msgid
失败:-1
//创建或打开消息队列
int msgid = msgget(key, IPC_CREAT | IPC_EXCL | 0666);
if (msgid <= 0)
{
if (errno == EEXIST)
{
msgid = msgget(key, 0666);
}
else
{
perror("msgget err");
return -1;
}
}
printf("msgid:%d\n", msgid);
2)添加消息msgsnd
int msgsnd(int msqid,constvoid*msgp, size_t size,int flag);
功能:
添加消息
参数:
msqid:消息队列的ID
msgp:指向消息的指针。
size:发送的消息正文的字节数
flag:IPC_NOWAIT消息没有发送完成函数也会立即返回
0:直到发送完成函数才返回
返回值:
成功:0
失败:-1
常用消息结构msgbuf如下:
struct msgbuf{
long mtype;//消息类型
char mtext[N]//消息正文
};
使用:msgsnd(msgid,&msg,sizeof(msg)-sizeof(long),0);
注意:消息结构除了第一个成员必须为long类型外,其他成员可以根据应用的需求自行定义。
struct mymsg
{
long type;
int num;
char buf[32];
};
//
//添加消息
struct mymsg msg ;
msg.type = 1;
strcpy(msg.buf, "hello");
msg.num = 23031;
msgsnd(msgid, &msg, sizeof(msg) - sizeof(long), 0);
3)读取消息msgrcv
int msgrcv(int msgid,void* msgp, size_t size,long msgtype,int flag);
功能:
读取消息
参数:
msgid:消息队列的ID
msgp:存放读取消息的空间
size:接受的消息正文的字节数
msgtype:
0:接收消息队列中第一个消息。
大于0:接收消息队列中第一个类型为msgtyp的消息.
小于0:接收消息队列中类型值不小于msgtyp的绝对值且类型值又最小的消息。
flag:0:若无消息函数会一直阻塞
IPC_NOWAIT:若没有消息,进程会立即返回ENOMSG
返回值:
成功:接收到的消息的长度
失败:-1
//接收消息
struct mymsg msg_r;
msgrcv(msgid, &msg_r, sizeof(msg_r) - sizeof(long), 1, 0);
printf("num:%d\n", msg_r.num);
printf("buf:%s\n", msg_r.buf);
4)删除消息队列msgctl
int msgctl(int msgqid,int cmd,structmsqid_ds*buf );
功能:
对消息队列的操作,删除消息队列
参数:
msqid:消息队列的队列ID
cmd:
IPC_STAT:读取消息队列的属性,并将其保存在buf指向的缓冲区中。
IPC_SET:设置消息队列的属性。这个值取自buf参数。
IPC_RMID:从系统中删除消息队列。
buf:消息队列缓冲区
返回值:
成功:0
失败:-1
用法:msgctl (msgid, IPC_RMID,NULL);
4. 命令
ipcs -q:查看消息队列
ipcrm -q msgid:删除编号为msgid的消息队列
如有错误,欢迎指正,谢谢。