访问共享内存中的结构后的Segfault?

时间:2013-04-20 17:48:30

标签: c struct posix shared-memory

我遇到了跨越进程共享存储在struct中的值的问题。在我的代码下面只使用一个进程进行了简化,这将增加值num2。每当流程结束时,waitpid()会将流程的pid写入 array 。这又简化了,在我更大的项目中,我有大约100个进程,它们在数组中连续编写了pid s。因此,每个进程都会看到数组。但是,对于它们中的每一个,struct中的整数值都不同。为什么?

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <semaphore.h>
#include <sys/shm.h>

typedef struct{
   int num;
   int num1;
   int num2;
   char *array;
} data;

void c_print(data *a);

int main(int argc, char *argv[])
{
   data *main_data;

   int pam_id=shmget(IPC_PRIVATE,sizeof(data), IPC_CREAT | IPC_EXCL | 0666);
   if (pam_id == -1)
       fprintf(stderr,"error");
   int k=shmat(pam_id,NULL,0);
   if (k==NULL)
       fprintf(stderr,"shmat error");
   main_data=malloc(sizeof(data));
   main_data->num = strtol(argv[1],NULL,10);
   main_data->num1 = strtol(argv[2],NULL,10);
   main_data->num2 = strtol(argv[3],NULL,10);

如果没有malloc,则访问main_data会导致 segfault 。但是,除了数组之外,其他进程无法查看struct中存储的变量。

  main_data->array = malloc(main_data->num2*sizeof(char));

  main_data->array[0]=fork();
  if (main_data->array[0]==0){
      main_data->num2+=2;
      exit(9);
  } else {
      waitpid(-1,main_data->array[0],0);
      c_print(main_data);
      return 0;
  }
  return 50;
}

1 个答案:

答案 0 :(得分:0)

您应该使用k而不是malloc()k的地址在进程之间共享。我认为你误解了 shm api。即,将main_data->array=...;替换为main_data->array=(void*)k;每个流程都会获得main_data的新副本。您需要将shmgetnum2*sizeof(char)一起使用。

您希望分享的内存应使用shmget()shmat()。每个进程将分配malloc(),并将使用父进程内存的副本进行初始化。如果numnum1num2为私有或const,则您的data结构可以使用malloc(),并且数组应使用shget 1}}和shmat。如果您需要在所有流程中动态更新两者,则两者都必须使用shget()shmat()

此外,您应该处理数据竞赛。