我的代码遇到了一些麻烦。它应该创建一个消息队列并发送消息,而不是等待一段时间让另一个程序接收该消息并回答。问题是,当我运行它时,我在msgsnd和msgrcv上都得到一个无效的参数。
#include <stdlib.h>
#include <unistd.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/msg.h>
typedef struct my_msg{
long type;
char text[100];
char sqdr;
}message;
static void score(int messagge_id, char* A_B){
message send;
send.type=1;
strcpy(send.text, "Try to score");
send.sqdr = *A_B;
if((msgsnd(messagge_id, &send, sizeof(send), 0))<0)perror("Error msgsnd\n");
sleep(3);
if((msgrcv(messagge_id, &send, sizeof(send), 4, 0))==-1)perror("Error msgrcv 1\n");
int test=atoi(send.text);
printf("%d\n", test);
}
int main(){
int caso, key;
char team= 'A';
key=1234;
int msg_id=msgget(key, S_IRUSR|S_IWUSR);
printf("Try function score\n");
score(msg_id, &team);
printf("After score\n");
return 0;
}
答案 0 :(得分:0)
您需要确保已创建消息队列。您可以使用密钥IPC_PRIVATE
,也可以将IPC_CREAT
添加到标记中。您还需要尝试正确阅读邮件。您发送了“类型1”消息并尝试读取“类型4”消息,因此读取挂起。
此代码还会删除邮件队列。如果它是一个私有队列(在程序终止时删除这样的队列),这并不重要,但对于使用IPC_CREAT
和用户定义密钥的队列来说这很重要。 (我还更改了消息文本,以便atoi()
返回更有趣的东西 - 并且令人信服 - 而不是零。代码还使用单独的发送和接收缓冲区,以便我们知道代码不会作弊并重新使用缓冲区中已有的数据。)
#include <stdlib.h>
#include <unistd.h>
#include <stdio.h>
#include <string.h>
#include <sys/msg.h>
#include <fcntl.h>
typedef struct my_msg
{
long type;
char text[100];
char sqdr;
} message;
static void score(int messagge_id)
{
message send;
message recv;
send.type = 1;
strcpy(send.text, "47 tries to score");
send.sqdr = 'A';
if ((msgsnd(messagge_id, &send, sizeof(send), 0)) < 0)
perror("Error msgsnd");
printf("Dozing...\n");
sleep(3);
printf("Unslumbering...\n");
if ((msgrcv(messagge_id, &recv, sizeof(recv), -4, 0)) == -1)
perror("Error msgrcv");
int test = atoi(recv.text);
printf("%d\n", test);
}
int main(void)
{
int key = 1234;
int flags = S_IRUSR|S_IWUSR|IPC_CREAT;
// int key = IPC_PRIVATE;
// int flags = S_IRUSR|S_IWUSR;
int msg_id = msgget(key, flags);
if (msg_id < 0)
perror("Error msgget");
else
{
printf("Try function score\n");
score(msg_id);
printf("After score\n");
if (msgctl(msg_id, IPC_RMID, 0) < 0)
perror("Error msgctl");
}
return 0;
}
示例输出:
Try function score
Dozing...
Unslumbering...
47
After score
当然,'Dozing'和'Unslumbering'之间有3秒的停顿。
答案 1 :(得分:-1)
像这样使用: -
if((msgsnd(messagge_id, (void *)&send, sizeof(send), 0))<0)perror("Error msgsnd\n");
if((msgrcv(messagge_id, (void *)&send, sizeof(send), 4, 0))==-1)perror("Error msgrcv 1\n");