【问题标题】:Message queue is giving me an invalid argument消息队列给了我一个无效的参数
【发布时间】:2016-12-24 13:46:16
【问题描述】:

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

【问题讨论】:

  • 为什么不对msgget() 调用进行错误检查?它成功了吗?是否有其他进程创建了消息队列?
  • 我会尝试检查 msgget,但我认为问题不在于消息队列的创建,因为我还没有编写第二个程序。会不会是使用的密钥的问题?
  • 我认为这很可能是您没有使用密钥IPC_PRIVATE 或指定IPC_CREAT 作为标志(连同S_IRUSR|S_IWUSR)的问题,所以您没有附加到一个预先存在的消息队列,也没有创建它。请参阅 msgget()(和 msgctl() 以在完成后删除消息队列)。请注意,一旦队列被创建,它会一直存在直到被删除。

标签: c


【解决方案1】:

您需要确保已创建消息队列。您要么使用密钥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

当然,在“打瞌睡”和“不眠不休”之间会有 3 秒的停顿。

【讨论】:

    【解决方案2】:

    这样使用:-

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

    【讨论】:

    • 有什么重大变化? void * 的演员表无关紧要;它没有任何改变。
    • 输出为“Try function score. Error msgsnd: invalid argument. Error msgrcv: invalid argument. After score”。
    猜你喜欢
    • 2017-08-25
    • 2011-07-10
    • 2012-12-05
    • 2021-10-14
    • 2012-05-24
    • 1970-01-01
    • 2015-12-17
    • 1970-01-01
    • 2012-08-29
    相关资源
    最近更新 更多