void test(char *p)
{
p = "666";
}
int main()
{
char *p = "test";
test(p);
printf("%s\n",p);
return 0;
}
以上代码指针P本来指向的是字符常量区的test的空间,现在把P这个指针的地址作为实参传递给子函数,在子函数中让P指向另外一段空间,那么main函数中再输出P的值是多少呢?
结果为:
test
为什么P不会被修改呢?
这里先简单穿插一个二级指针的概念:
(什么是二级指针,顾名思义就是以一个存放指针的指针)
int a = 10;
int *p = &a;
int **p1 = &p;
printf("a is %p\n",&a);
printf("p is %p\n",p);
printf("%p\n",&p);
printf("%p\n",p1);
printf("p1 %p\n",&p1);
结果为:
a is 0x7ffcdfe2e444
p is 0x7ffcdfe2e444
0x7ffcdfe2e448
0x7ffcdfe2e448
p1 0x7ffcdfe2e450
由此我们可以得知
回到上面的问题,为什么没有修改成功呢?
①原因是我们传递的P并不是指针的地址,而是指针里存放的字符串常量的地址。
②并且子函数中的赋值并不是修改了test这个内存中的内容,而是让形参的指针指向了另一块地址,所以从始至终main函数中P指向的地址和内容始终没有变化,这也就造成了我们修改内容不成功。
想修改应该用二级指针直接传递P的地址。
void test(char **p)
{
*p = "666";
}
int main()
{
char *p = "test";
test(&p);
printf("%s\n",p);
return 0;
}
上图代码可以用下图来表示
当我们传递的是P这个指针本身的地址时,他就是一个二级指针,因为它内部还存了一个地址,而*P取的就是二级指针中存放的那个地址。我们现在让P指针里存放的地址变成存放666的那块内存。这样一来,main函数中P指针存放的地址就会改变。实现内容修改
以上都是针对字符串常量,因为字符串常量的值不可更改,如果指针存放的是整形这些可以修改的值,就很方便,比如以下代码
void test(int *p)
{
*p = 100;
}
int main()
{
int a = 10;
test(&a);
printf("%d\n",a);
return 0;
}
如上所示,我们可以通过指针直接修改指针指向的内容。