引言
最近碰到几道很有必要一提的试题,这里记录一下,主要是关于指针的一些理解。
实例
下面将试题附上:
例1
void GetMemory(char **p,int num)
{
*p = (char*)malloc(num);
}
void test()
{
char *str = NULL;
GetMemory(&str,100);
strcpy(str,"hello");
printf(str);
}
会问输出为什么结果。正确的答案是hello,这道题主要考查的是指针作为参数的传递,也是二级指针的理解,这里直接将指针str的地址传递到GetMemory函数中,故而申请了100个字节的内存,让str指针指向这块内存区域,然后往str指针指向的内存写入字符串hello,最后输出str指向的内存的内容hello。
例2
void test1()
{
char *str = (char*)malloc(100);
strcpy(str,"hello");
free(str);
if(str != NULL)
{
strcpy(str,"world");//str已经变为悬垂指针,依旧指向原来的位置,但是原来的内存已经不受保护
printf(str);
}
}
这段代码,起初分配了100字节的内存空间,用指针str指向这块内存,向内存中写入hello,但是又调用free释放掉了str指向的内存,没有给str赋空,因此str指针依旧指向那块内存,只是这块内存不再被管理,当调用strcpy函数向str指向的内存写入world的时候,就相当于给悬垂指针指向的内存区域写入字符串,虽然程序最后输出world但是不能正常运行,会出现程序崩溃,究其原因是因为给悬垂指针指向的内存写入字符串所致。
以上要是想改正,可如下修改:
void test5()
{
char *str = (char*)malloc(100);
strcpy(str,"hello");
free(str);
str = NULL;
str = (char*)malloc(100);
if(str != NULL)
{
strcpy(str,"world");//str已经变为悬垂指针,依旧指向原来的位置,但是原来的内存已经不受保护
printf(str);
}
}
这里只做了大致的修改,一些判空操作被省略。
例3
void getMemory(char *p)
{
p = (char *)malloc(100);
}
void Test()
{
char *str = NULL;
getMemory(str);
strcpy(str,"hello world");
printf(str);
}
此例会导致程序运行崩溃,在调用函数getMemory之后,并没有将str指向某一块固定大小的内存,因为只将str传递给临时指针p,只是将临时变量p指向了一块100字节的内存,而str则依旧为null,所以接下来调用函数strcpy给str指向的内存赋值hello world的时候,会导致程序的崩溃,没有输出。
例4
char *getMem()
{
char p[] = "hello world";//栈区
return p;
}
void test2()
{
char *str = NULL;
str = getMem();
printf(str);
}
这段程序输出乱码。当调用函数getMem返回指针p指向的内存地址,但是由于p是局部变量,又存储在栈区,故这块内存区域被释放,str被赋值指针p指向的地址返回值后,str指向这块被释放的内存,当输出str指向的内存的内容时就会出现乱码。
下面就例4进行修改和试验:
char *getMem()
{
char* p = "hello world";//常量区?
printf("-->%s,0X=%p,0x=%p\n",p,&p,p);
return p;
}
void test2()
{
char *str = NULL;
str = getMem();
printf("==>%s, 0x=%p",str,str);
}
将数组char p[] = “hello world”;改为指针指向字符串常量后,即char* p = “hello world”;,字符串常量存储在常量区,即使返回临时变量p指向的内存地址,局部变量p被回收,常量区的数据hello world依旧存在,str指向常量hello world所在的内存区域,输出str的时候输出hello world.
今天发现之前的草稿箱中存了一些之前写的文章,就直接发布了,应该当时是验证过得。