消息队列给我一个无效的参数

时间:2016-12-24 13:46:17

标签: c

我的代码遇到了一些麻烦。它应该创建一个消息队列并发送消息,而不是等待一段时间让另一个程序接收该消息并回答。问题是,当我运行它时,我在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;
}

2 个答案:

答案 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");