这个周的Linux实验是做基于消息队列和共享内存的进程间通信,要求主要如下:
1)消息的创建、发送和接收
使用消息调用msgget()、msgsnd()、msggrev()msgctrl()编制长度为1K的消息的发送和接收程序。
2)共享存储取得创建、附接和断接
使用系统调用shmget()、shmat()、shmctl()、shmctl(),编制一个与上述功能相同的程序。
这就需要将基于消息队列的和基于共享内存的程序分开写两个
程序的要求
为了便于操作和观察结果,用一个程序作为“引子”,先后fork()两个子进程,server和client,进行通信;
server端建立一个key为75的消息队列,等待其他进程发来的消息。当遇到类型为1的消息时,则作为结束信号,取消该队列,并退出Server。Server每收到一个消息后显示一句“(Server)received”;
Client端使用key为75的消息队列,先后发送类型为10到1的消息,然后退出。最后一个消息即server端需要的结束信号。Client每发送一条消息,显示一句“(Client)sent”;
父进程在server和client均退出后结束。
要求解析
key为75,就是描述符为75.
客户端要发送十个消息,则用for循环比较容易实现
比如这样形式
for(i=10;i>=1;i--)
我写头文件就比较喜欢把所有这方面的都加上,
#include <sys/types.h>
#include <sys/msg.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/wait.h>
#include <string.h>
#include <strings.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
程序主要是由一个主进程和客户端加服务端来组成,准确的说应该是主进程来调用客户端和服务端。
代码示例
基于消息队列
#include <sys/types.h>
#include <sys/msg.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/wait.h>
#include <string.h>
#include <strings.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
struct zbxzbx
{
long m;
char n[100]; /*文本长度*/
}msg;
int msgqid,i;
void CLIENT( )
{
int i;
msgqid=msgget(75,0666|IPC_CREAT);
for(i=10;i>=1;i--)
{
msg.m=i;
printf("(client)sent\n");
msgsnd(msgqid,&msg,1030,0); /*发送消息msg入msgid消息队列*/
}
exit(0);
}
void SERVER( )
{
msgqid=msgget(75,0666|IPC_CREAT); /*由关键字获得消息队列*/
do
{
msgrcv(msgqid,&msg,1030,0,0); /*从队列msgid接受消息msg*/
printf("(server)receive\n");
}while(msg.m!=1); /*消息类型为1时,释放队列*/
msgctl(msgqid, IPC_RMID,0);
}
int main()
{
if(fork())
{
SERVER();
wait(0);
}
else CLIENT( );
}
基于共享内存
#include <sys/types.h>
#include <sys/msg.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/wait.h>
#include <string.h>
#include <strings.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
int shmid,i;
int *addr;
int CLIENT()
{
int i;
shmid=shmget(75,1024, 0777|IPC_CREAT); /*获取共享区,长度1024/
addr=shmat(shmid,0,0); /*共享区起始地址为addr*/
for(i=9;i>=0;i--)
{
while(*addr != -1);
printf("(client)sent\n"); /*打印(client)sent*/
*addr=i; /*把i赋给addr*/
}
exit(0);
}
int SERVER()
{
do{
while(*addr == -1);
printf("(server)received\n%d",*addr); /*服务进程使用共享区*/
if(*addr!=0)
*addr=-1;
}while(*addr);
wait(0);
shmctl(shmid,IPC_RMID,0);
}
int main()
{
shmid=shmget(75,1024,0777|IPC_CREAT); /*创建共享区*/
addr=shmat(shmid,0,0); /*共享区起始地址为addr*/
*addr=-1;
if(fork())
{
SERVER();
}
else
{
CLIENT();
}
}
效果图