需求:
1:有一块共享内存,分为四个缓存区,2:每一个缓存区又是一个数组,3:每一个数组里面存放着结构体指针
如图
遇到的问题:不能成功申请内存
希望高手指点一下迷津
代码如下:
#include
#include
#include
#include
#include
#include
typedef struct {
int32_t a;
int64_t b;
int64_t c;
}Small;
typedef struct {
int index[4];
int small_count[4];
Small *small[4][20];
}Big;
int main(){
Big *big = NULL;
int shm_id = shmget((key_t)12345,sizeof(Big),0666 | IPC_CREAT);
if(shm_id < 0){
printf("get shm_id errorn");
return -1;
}
void *shm_pointer = shmat(shm_id,(void*)0,0);
if(shm_pointer == NULL){
printf("shmmat errorn");
return -1;
}
big = (Big*)shm_pointer;
printf("big=%pn",big);
printf("big->small[0]=%pn",big->small[0]);
printf("big->small[0][0]=%pn",big->small[0][0]);
//next step: I want to memcpy data to the shared_memory
Small small;
small.a = 1;
small.b = 1;
small.c = 1;
memcpy(big->small[2][10],&small,sizeof(Small));
return 0;
}
回答
你看一下
sizeof(Small)
sizeof(Big)
就应该知道为什么不对了。
sizeof(Big)的大小可不是你预期的大小
4*sizeof(int) + sizeof(int)*4 + 4*20*sizeof(Small)
而是
4*sizeof(int) + sizeof(int)*4 + 4*20*sizeof(Small*)
sizeof(Small) = sizeof(int32) + sizeof(int64)*2 (在不考虑机器字节对齐的情况下)
而sizeof(Small*) = sizeof(void*)
就是机器指针长度一般为 4
big->small[0][0]相当于初始化的一个指针变量 默认初始化为0
所有big->small[X][X] 都被赋值为0
除非你申请个堆空间给他赋值
后面这个句子就不会段错误
memcpy(big->small[2][10],&small,sizeof(Small));
还有比较简单的改法就是结构体Big定义的时候把Small * 的这个* 去掉
变为
typedef struct {
int index[4];
int small_count[4];
Small small[4][20];
}Big;
即可符合你的预期