一开始没有理解,导致对这个概念非常的模糊和不解。最近看完《 彻底搞定C语言指针详解》,里面关于指针的指针的解释有了顿悟感。
其实,很好理解,是我把它想复杂了。
我们都知道,实参和形参的区别。形参并无法改变实参,这就是值传递。还有一种叫指针传递,实质就是去操作指针的指向对象。例子如下:
Exchg2(int *px, int *py)
{
int tmp=*px;
*px=*py;
*py=tmp;
print(“*px=%d,*py=%d\n”,*px,*py);
}
main()
{
int a=4;
int b=6;
Exchg2( &a,&b);
Print (“a=%d,b=%d\n”, a, b);
}
这时候打印出来的结果是什么呢?
a=6,b=4 px = 6, py =4
为什么?
Exchg2这个函数的参数是两个int型指针,是地址,所以你要用&取a和b的地址传入。在Exchg2的函数内部,两个指针所指向的值交换了,因为之前你已经把a和b的地址传递给了px和py这两个指针,所以现在操作px和py指向的值,也会导致a和b所指向的值发生变化。
这个虽然饶了些,但还是勉强能理解。那么,如果是指针的指针呢?
如下:
这个函数本意是为传入的字符串开辟空间,但是仔细看,它真的达到了这个效果吗?
void GetMemory(char *p, int num)
{
p = (char *)malloc(sizeof(char) * num);
}
main(int argc, _TCHAR* argv[])
{
char *str = NULL;
GetMeory(str, 100);
strcpy(str,"Hello");
cout << str << endl;
return 0;
}
显然不能,为什么?这其实和值传递是一个意思,你传递进来的是一个指针实参str,而函数的形参p去替换了实参,所以后面操作的都是形参而已,并不会对实参有影响。这就不符合我们要求了啊,那怎么办?上一个例子,我们传入实参的地址,然后交换地址指向的值,来达到我们交换实参的目的,那么在这边我们也可以这么干啊。
void GetMeory(char **p, int num)
{
*p = (char *)malloc(sizeof(char) * num);
//*p = new char[num]; //C++当中
}
int _tmain(int argc, _TCHAR* argv[])
{
char *str = NULL;
GetMeory(&str, 100);
strcpy(str,"Hello");
cout << str << endl;
return 0;
}
这不就是把str的地址&str传入了GetMeory函数内嘛,只不过这个str本身也是一个指针,管它呢,还有就是GetMeory中的参数变成了char **p,这也很好理解,我传入的是指针的指针,如果你还是只用指针char *p来接收的话,不就也只是操作形参了和值传递没区别了啊。我要操作的是实参指针,所以形参必须是实参的指针,也就是char **, 好绕口,不过 不难理解,你想想为什么值传递没有改变实参就知道了。
最后我们改变的是p,也就是形参char p的值,因为是取值符。指的是p指向的值,也就是一个地址,是指针。和指针传递的那个例子不同,那个例子是为了交换两者指向的值,所以要看你想实现什么,来决定操作什么值。