Compare commits

..

2 Commits

Author SHA1 Message Date
flykhan e6757b9935 消息队列: 发送者接受者 2023-08-18 15:05:49 +08:00
flykhan c23d7e1ccc 消息队列案例: 单机聊天程序 2023-08-18 15:03:56 +08:00
7 changed files with 240 additions and 0 deletions

BIN
day5/chat_tool/bob Executable file

Binary file not shown.

View File

@ -0,0 +1,88 @@
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <string.h>
#include <stdlib.h>
#include <time.h>
#include <unistd.h>
#include <sys/wait.h>
#include <signal.h>
typedef struct msg_
{
long mType; // 消息类型
char content[100]; // 正文数据
char name[32]; // 姓名
} MSG;
int main(int argc, char const *argv[])
{
if (argc != 2)
{
printf("invalid, format is : ./xx name\n");
return 1;
}
key_t key = ftok("/", 160);
int msgqid = msgget(key, IPC_CREAT | 0666);
if (msgqid == -1)
{
perror("msgget");
return 1;
}
int pid = fork();
if (pid == 0)
{
// 子进程
while (1)
{
write(STDOUT_FILENO, "我说:", 7);
MSG msg;
scanf("%s", msg.content);
strcpy(msg.name, argv[1]);
#ifdef BOB
msg.mType = 1;
#else
msg.mType = 2;
#endif
if (msgsnd(msgqid, &msg, sizeof(MSG) - sizeof(long), 0) != -1)
{
printf("send msg OK\n");
if (strncmp(msg.content, "bye", 3) == 0)
{
break;
}
}
}
// kill(getppid(), SIGSTOP);
_exit(0);
}
else if (pid > 0)
{
// 父进程
while (1)
{
MSG msg;
#ifdef BOB
msg.mType = 2;
#else
msg.mType = 1;
#endif
if (msgrcv(msgqid, &msg, sizeof(MSG) - sizeof(long), msg.mType, 0) != -1)
{
printf("\n%s >>> \"%s\"\n", msg.name, msg.content);
if (strncmp(msg.content, "bye", 3) == 0)
{
break;
}
write(STDOUT_FILENO, "我说:", 7);
}
}
// kill(pid, SIGSTOP);
wait(NULL);
}
return 0;
}

BIN
day5/chat_tool/lucy Executable file

Binary file not shown.

View File

@ -0,0 +1,90 @@
#include <stdio.h>
#include <sys/types.h> // 系统数据类型定义
#include <sys/ipc.h> // IPC相关头文件
#include <sys/msg.h> // 消息队列相关头文件
#include <string.h> // 字符串操作
#include <stdlib.h> // atol 函数
// 消息结构体
typedef struct msg_
{
long mType; // 消息类型,必须是 long ,必须在第一个位置
char content[100]; // 消息内容,需要小于消息队列的限制值(8192)
char title[32]; // 消息标题
} MSG;
// msgget 函数用于创建消息队列参数1是键值参数2是权限
int main(int argc, char *argv[])
{
// 判断参数个数
if (argc != 4)
{
printf("Usage: %s <type> <content> <title>\n", argv[0]);
return 1;
}
key_t key = ftok("/", 200); // 获取消息队列的键值
// msgget 函数用于创建消息队列参数1是键值参数2是权限
// 返回值是消息队列的标识符,类似于文件描述符
// IPC_CREAT 表示如果消息队列不存在则创建,如果存在则打开
// 0644 表示权限,类似于文件权限,表示所有者可读写,其他人只读
int msgqid = msgget(key, IPC_CREAT | 0666); // 创建消息队列
if (msgqid == -1)
{
perror("msgget");
return 1;
}
// msgqid 只要不是 -1 就表示创建成功
printf("msgqid = %d\n", msgqid); // 打印消息队列标识符
// 发送消息
// 1 表示类型,表示发送消息的类型,必须大于 0
// MSG msg1 = {1, "测试消息队列的发送函数 msgsnd", "测试"}; // 消息内容
MSG msg1;
msg1.mType = atol(argv[1]); // 消息类型
strcpy(msg1.content, argv[2]); // 消息内容
strcpy(msg1.title, argv[3]); // 消息标题
// 参数 4 表示阻塞,如果消息队列满了,发送消息的进程会阻塞
// 非阻塞的话,消息队列满了,发送消息的进程会立即返回,不会阻塞
// int ret = msgsnd(msgqid, &msg1, sizeof(msg1.content), 0); // 发送消息
int ret = msgsnd(msgqid, &msg1, sizeof(msg1.content) + sizeof(msg1.title), IPC_NOWAIT); // 发送消息
if (ret == -1)
{
perror("msgsnd");
return 1;
}
printf("msg1 发送成功\n");
return 0;
}
/*
ipc: inter process communication
ipcs -q // 查看消息队列
ipcrm -q <msqid> // 删除消息队列
*/
/*
ipc
1.
2.
3.
4.
5.
*/
/*
使 msgque3_receiver.c (receiver) 使:
1. msgque2_sender.c (sender) msgque3_receiver.c (receiver)
2. msgque2_sender.c (sender)
3. msgque3_receiver.c (receiver)
ipcs -q
ipcrm -q <msqid>
*/

View File

@ -0,0 +1,62 @@
#include <stdio.h>
#include <sys/types.h> // 系统数据类型定义
#include <sys/ipc.h> // IPC相关头文件
#include <sys/msg.h> // 消息队列相关头文件
#include <string.h> // 字符串操作
#include <stdlib.h> // atol 函数
// 消息结构体
typedef struct msg_
{
long mType; // 消息类型,必须是 long ,必须在第一个位置
char content[100]; // 消息内容,需要小于消息队列的限制值(8192)
char title[32]; // 消息标题
} MSG;
// msgget 函数用于创建消息队列参数1是键值参数2是权限
int main(int argc, char *argv[])
{
// 参数个数判断
if (argc != 2)
{
printf("Usage: %s <type>\n", argv[0]);
return 1;
}
key_t key = ftok("/", 200); // 获取消息队列的键值
int msgqid = msgget(key, IPC_CREAT | 0666); // 创建消息队列
if (msgqid == -1)
{
perror("msgget");
return 1;
}
// 接收消息
MSG msg; // 消息结构体
long mtype = atol(argv[1]); // 消息类型
// 最后参数 0 表示阻塞,如果消息队列为空,接收消息的进程会阻塞
ssize_t len = msgrcv(msgqid, &msg, sizeof(msg.content) + sizeof(msg.title), mtype, 0); // 接收消息
printf("接收到的数据(%ld)\n类型(%ld)\n内容(%s)\n标题(%s)\n", len, msg.mType, msg.content, msg.title); // 打印消息
return 0;
}
/*
使 msgque2_sender.c (sender) 使:
1. msgque2_sender.c (sender) msgque3_receiver.c (receiver)
2. msgque2_sender.c (sender)
3. msgque3_receiver.c (receiver)
使:
./receiver <type>
type 0
type -5 5
type 1 1
type 0
ipcs -q
ipcrm -q <msqid>
*/

BIN
day5/rec_sed/receiver Executable file

Binary file not shown.

BIN
day5/rec_sed/send Executable file

Binary file not shown.