下面一段关于GetMemory()的代码是面试时最容易出现的题目。
发现自己在解释原理的时候,还是有点“拗口”, 就来总结一下吧。
下例实际上涉及的就是传参的方式或堆内存的传递(堆内存靠指针来传递)。
传参的方式:
1.值
2.指针(如果变量本身就是一个指针,把此变量当作一个对象,就要传指针的指针?)
3.引用(实际就是传变量本身,不管变量是一个对象还是一个指针)
其它类似Topic,参考“ 局部变量返回值”。
发现自己在解释原理的时候,还是有点“拗口”, 就来总结一下吧。
下例实际上涉及的就是传参的方式或堆内存的传递(堆内存靠指针来传递)。
传参的方式:
1.值
2.指针(如果变量本身就是一个指针,把此变量当作一个对象,就要传指针的指针?)
3.引用(实际就是传变量本身,不管变量是一个对象还是一个指针)
详见代码注释。
#include <iostream>
using namespace std;
//之前我对这里的疑惑:
//传指针不是不会拷贝副本吗?(这里把指针也当对象用?)
//当传入str时,str与p值一样,指向也一样,当p改变指向时,str也应该改变指向?
void GetMemory(char *p)
{
// p是str(实参)的拷贝,它们的值相同,但是地址不同。
// 如果实参本身是一个对象(传值),这里就是这个对象的指针
cout<<p<<endl;
//p指向另外一块内存,而str的指向不变。
p = (char *)malloc(100);
}
void GetMemory1(char* &p)
{
//指针的引用,p和str(实参)为同一个变量
p = (char *)malloc(100);
}
void GetMemory2(char **p)
{
//p是&str(实参)的拷贝,它们的值相同,但是地址不同。
//但是*p和str是同一个变量(值相同,地址相同(p与&str的值)),所以*p改变指向,意味着str也改变指向
*p = (char *)malloc(100);
}
//返回值传递堆内存
char* GetMemory3()
{
char *p = (char *)malloc(100);
}
void Test1(void)
{
char *str = "test\n";
//这里str指向是常量,str并没有获取到堆内存,再改变常量的值,出现segmentation fault
GetMemory(str);
//传指针的引用,OK
//GetMemory1(str);
strcpy(str, "hello world");
}
void Test2(void)
{
char *str = "test\n";
GetMemory2(&str);
//这里&str的值是不会变的,但是str的值发生了改变(获取到堆内存)。
strcpy(str, "hello\n");
printf(str);
free((void*)str);
}
void Test3(void)
{
//str和GetMemory3()中的临时变量p指向同一块内存(p销毁,str还在,内存还在)
char *str = GetMemory3();
strcpy(str, "hello world\n");
printf(str);
free(str);
}
int main(void)
{
Test2();
Test3();
Test1();
}
其它类似Topic,参考“ 局部变量返回值”。