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