试题4:
void GetMemory( char *p ) { p = (char *) malloc( 100 ); }
void Test( void ) { char *str = NULL; GetMemory( str ); strcpy( str, "hello world" ); printf( str ); } |
试题5:
char *GetMemory( void ) { char p[] = "hello world"; return p; }
void Test( void ) { char *str = NULL; str = GetMemory(); printf( str ); } |
试题6:
void GetMemory( char **p, int num ) { *p = (char *) malloc( num ); }
void Test( void ) { char *str = NULL; GetMemory( &str, 100 ); strcpy( str, "hello" ); printf( str ); } |
试题7:
void Test( void ) { char *str = (char *) malloc( 100 ); strcpy( str, "hello" ); free( str ); ... //省略的其它语句 } |
解答:
试题4传入中GetMemory( char *p )函数的形参为字符串指针,在函数内部修改形参并不能真正的改变传入形参的值,执行完
char *str = NULL; GetMemory( str ); |
后的str仍然为NULL; ------------------------------------------------------------------------- 指针传递还是按值传递 所以并没有实现帮你把str开一块空间 只是值的拷贝,就是把str的值赋给p 然后p开了空间,但是str还是没开 所以你要void GetMemory(char **p, int num) *p = (char *)malloc(sizeof(char) * num); char *str = NULL; GetMemory(&str, 100); ------------------------------------------------------------------------- 试题5中
char p[] = "hello world"; return p; |
的p[]数组为函数内的局部自动变量,在函数返回后,内存已经被释放。这是许多程序员常犯的错误,其根源在于不理解变量的生存期。
试题6的GetMemory避免了试题4的问题,传入GetMemory的参数为字符串指针的指针,但是在GetMemory中执行申请内存及赋值语句
*p = (char *) malloc( num ); |
后未判断内存是否申请成功,应加上:
if ( *p == NULL ) { ...//进行申请内存失败处理 } |
试题7存在与试题6同样的问题,在执行
char *str = (char *) malloc(100); |
后未进行内存是否申请成功的判断;另外,在free(str)后未置str为空,导致可能变成一个“野”指针,应加上:
试题6的Test函数中也未对malloc的内存进行释放。
剖析:
试题4~7考查面试者对内存操作的理解程度,基本功扎实的面试者一般都能正确的回答其中50~60的错误。但是要完全解答正确,却也绝非易事。
对内存操作的考查主要集中在:
(1)指针的理解;
(2)变量的生存期及作用范围;
(3)良好的动态内存申请和释放习惯。
再看看下面的一段程序有什么错误:
swap( int* p1,int* p2 ) { int *p; *p = *p1; *p1 = *p2; *p2 = *p; } |
在swap函数中,p是一个“野”指针,有可能指向系统区,导致程序运行的崩溃。在VC++中DEBUG运行时提示错误“Access Violation”。该程序应该改为:
swap( int* p1,int* p2 ) { int p; p = *p1; *p1 = *p2; *p2 = p; }
|
请发表评论