指针作为形参时,1、可以改变指针指向的内容,2、但是无法改变指针本身的地址,3、除非在改变指针地址时,采用返回值。
文章目录
##1、可以改变指针指向的内容
void test(int* s){
*s = 100;
}
int main(){
int a = 10;
int* p = &a;
cout << *p << endl;
test(p);
cout << *p << endl;
return 0;
}
执行过程: test()函数在调用时,生成临时变量int* s,s等于p,p用于s的初始化,由于s等于p,所以s和p指向的地址是相同的,通过s对地址上的内容进行改变,和通过p对地址上的内容进行改变是一样的!
##2、无法改变指针本身的地址
void test(int* s){
s = new int[1];
}
int main(){
int *p = NULL;
test(p);
if(p) delete[]p;
return 0;
}
重点:变量作为函数形参调用时,都会分配一个副本,不管是传值还是传址,传入之后就和形参没有关系了,不会改变形参的值。
执行过程:test()函数在调用时,会分配一个临时变量int* s,s等于p,p只是用于s的初始化,**此后就一点关系都没有了!**s = new int[1],对p没有任何影响,p的值还是NULL。
##3、改变指针地址时,采用返回值
char *GetMemory3(int num)
{
char *p = (char *)malloc(sizeof(char) * num);
return p;
}
void Test3(void)
{
char *str = NULL;
str = GetMemory3(100);
strcpy(str, "hello");
cout<< str << endl;
free(str);
}
但是不能用return语句返回“栈内存”的指针,因为该指针会在函数结束时,自动消亡
char *GetString(void)
{
char p[] = "hello world";
return p; // 编译器将提出警告
}
void Test4(void)
{
char *str = NULL;
str = GetString(); // str 的内容是垃圾
cout<< str << endl;
}
str = GetString语句后str不再是NULL指针,但是str的内容不是“hello world”而是垃圾。
##4、使用二级指针实现内存申请,通过指针值传递
void my_malloc1(char **p1)
{
*p1 = new char[100];
}
int main()
{
char *buffer = NULL;
my_malloc1(&buffer);
printf("buffer adress is %x/n", buffer);
free(buffer);
return 0;
}
##5、总结
1、一级指针和二级指针在做形参时的不同:指针用作形参,改变指针地址则值不能传回,改变指针内容而地址不变则值可以传回。(特殊情况:改变指针地址采用返回值也可以传回地址)
2、对于一级指针,做形参时传入地址,如果函数只改变该指针内容,OK,该指针可以正常返回,如果函数改变了指针地址,除非返回该指针,否则该指针不能正常返回,函数内对指针的操作将无效。
3、对于二级指针,做形参时传入地址(注意此时传入的是二级指针的地址),如果改变该二级指针地址(**p),对该指针的操作也将无效,但是改变二级指针的内容(例如*p),则该二级指针可以正常返回。
总之,指针使用最关键的是弄清地址和内容,指针做形参时只有改变其内容时才能正常返回。