将结构放入共享内存

时间:2015-11-22 18:24:21

标签: c linux struct shared-memory

我创建了两个程序server.c和client.c。我有一个持有年龄的结构。我让程序一起工作来读取共享内存并更改共享内存,但这仅在结构中使用一个变量时才有效。只要我在结构中有多个变量,就会出现分段错误。

Server.c

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>

typedef struct People
{
    int age;
    int isDone;

} Person;

int main()
{
    Person aaron;
    Person *p_aaron;
    int id;
    int key = 5432;

    p_aaron = &aaron;
    (*p_aaron).age = 19;
    (*p_aaron).isDone = 0;

    if ((id = shmget(key,sizeof(aaron), IPC_CREAT | 0666)) < 0)
    {
        perror("SHMGET");
        exit(1);
    }


    if((p_aaron = shmat(id, NULL, 0)) == (Person *) -1)
    {
        perror("SHMAT");
        exit(1);
    }

    (*p_aaron).age = 19;


    printf("Shared Memory Age: %d\n", (*p_aaron).age);
    *p_aaron = aaron;

    while ((*p_aaron).age == 19)
    {
        sleep(1);
    }

    printf("Shared Memory Age Turned To: %d", (*p_aaron).age);


    return 0;
}

Client.c

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdio.h>
#include <string.h>

typedef struct People
{
    int age;

} Person;

int main()
{
    Person aaron;
    Person *p_aaron;
    int id;
    int key = 5432;

    p_aaron = &aaron;

    id = shmget(key,sizeof(aaron), IPC_CREAT | 0644);
    p_aaron = shmat(id, NULL, 0);

    printf("%d", (*p_aaron).age);
    (*p_aaron).age = 21;


    return 0;
}

来自Server.c的错误消息

SHMGET: Invalid argument

RUN FINISHED; exit value 1; real time: 0ms; user: 0ms; system: 0ms

1 个答案:

答案 0 :(得分:1)

您不会显示任何删除共享内存段的代码。

如果查看shmget()的POSIX规范,您会看到在以下情况下可以提供您报告的EINVAL错误:

  

[EINVAL]
      将创建共享内存段,并且size的值小于系统强加的最小值或大于系统强加的最大值。
  [EINVAL]
      不会创建共享内存段,并且密钥存在共享内存段,但与其关联的段的大小小于大小。

我想你可能遇到了第二种情况;你试图创建一个比已经存在的更大的共享内存段。

  • 修改您的代码以便自我清理(shmdt()shmctl())。
  • 使用ipcrm删除现有的共享内存段。

另外,正如我在comment中所提到的,您应该确保客户端和服务器程序就共享内存中结构的大小达成一致。其他任何东西都是灾难的秘诀。您应该将结构定义放入标题中,并且您的程序都应该使用该标题,并且在更改标题的定义时都应重新编译它们。