在对应最近的项目的时候,遇到了一次需要用双重指针的对应的情况,这也是第一次在实战中运用双重指针,特此记录下来,以下文章中我都简单定义了一些结构体和函数来方便描述整个过程。
具体情景:
目前功能需要追加一个get接口,由另一个模块的线程(为了后续描述方便,这里我们称为threadA)来到我这个模块的线程(这里称为threadB)来get某项信息,get的对象类型是结构体Object,数量未知,所以内存的申请与释放都是我这边来处理。
typedef struct
{
char id;
int value;
}Object;
起初的操作是threadA声明了一个结构体变量 stObject,然后将stObject的地址传给threadB,threadB 去申请内存,进行变量的初始化与赋值,最后负责内存的释放。
下面写个简略代码,就不加多线程了,方便大家看一下:
#include <stdio.h>
#include <stdlib.h>
#include <memory.h>
typedef struct
{
char id;
int value;
}Object;
void getObj(Object *pData);
void getObj(Object *pData)
{
int num = 6;
if(NULL != pData)
{
pData = (Object *)malloc(num * sizeof(Object));
}
pData->id = 1;
printf("set id = %d\n", pData->id);
}
int main()
{
Object stObj;
memset(&stObj, 0x00, sizeof(stObj));
getObj(&stObj);
printf("obj id = %d\n", stObj.id);
return 0;
}
程序输出结果如下:
set id = 1
obj id = 0
可以看出,id的值并没有变化,因为这里忽略了一点,调用getObj的时候虽然传入的实参是stObj的地址,但是pData是形参,pData本身的地址和stObj的地址不同,pData申请到了内存,并且进行了赋值,但是stObj的值并没有改变。平时我们通过函数出参改变参数的值,改变的是指针指向的内容,而并不是指针本身,这里想要改变指针本身,所以才会发生错误。
那么有什么其他的方法能将上面这个问题实现呢?这里可以通过双重指针,来看下面这段简写的代码。
#include <stdio.h>
#include <stdlib.h>
#include <memory.h>
typedef struct
{
char id;
int value;
}Object;
void getObj(Object **pData);
void getObj(Object **pData)
{
int num = 6;
if(NULL != pData)
{
*pData = (Object *)malloc(num * sizeof(Object));
}
(*pData)->id = 1;
printf("set id = %d\n", (*pData)->id);
}
int main()
{
Object *pObj = NULL;
getObj(&pObj);
printf("obj id = %d\n", pObj->id);
return 0;
}
运行结果如下:
set id = 1
obj id = 1
想通过函数参数形式修改指针地址,一般使用二级指针。这里,将指针pObj的地址赋给了二级指针pData, 那么 * pData 指向的就是&pObj指向的内容,也就是pObj本身,所以在这里对 * pData申请内存、赋值,也就实现了对空指针pObj的内存分配和赋值。