分析這些面試題,本身包含很強的趣味性;找出其中的錯誤:
試題1:
void test1()
{ char string[10]; char* str1 = "0123456789"; strcpy( string, str1 ); } |
試題2:
void test2()
{ char string[10], str1[10]; int i; for(i=0; i<10; i++) { str1[i] = 'a'; } strcpy( string, str1 ); } |
試題3:
void test3(char* str1)
{ char string[10]; if( strlen( str1 ) <= 10 ) { strcpy( string, str1 ); } } |
解答:
試題1字串str1需要11個位元組才能存放下(包括末尾的‘\0’),而string只有10個位元組的空間,strcpy會導致數組越界;
對試題2,如果面試者指出字元數組str1不能在數組內結束可以給3分;如果面試者指出 strcpy(string, str1)調用使得從str1記憶體起複製到string記憶體起所複製的位元組數具有不確定性可以給7分,在此基礎上指出庫函數strcpy工作方式的給10 分;
對試題3,if(strlen(str1) <= 10)應改為if(strlen(str1) < 10),因為strlen的結果未統計‘\0’所佔用的1個位元組。
剖析:
考查對基本功的掌握:
(1)字串以‘\0’結尾;
(2)對數組越界把握的敏感度;
(3)庫函數strcpy的工作方式,如果編寫一個標準strcpy函數的總分值為10,下面給出幾個不同得分的答案:
2分 void strcpy( char *strDest, char *strSrc ) { while( (*strDest++ = * strSrc++) != ‘\0’ ); } 4分 void strcpy( char *strDest, const char *strSrc )
//將源字串加const,表明其為輸入參數,加2分 { while( (*strDest++ = * strSrc++) != ‘\0’ ); } 7分 void strcpy(char *strDest, const char *strSrc) { //對源地址和目的地址加非0斷言,加3分 assert( (strDest != NULL) && (strSrc != NULL) );
while( (*strDest++ = * strSrc++) != ‘\0’ ); } 10分 //為了實現鏈式操作,將目的地址返回,加3分! char * strcpy( char *strDest, const char *strSrc )
{ assert( (strDest != NULL) && (strSrc != NULL) ); char *address = strDest; while( (*strDest++ = * strSrc++) != ‘\0’ ); return address; } |
從2分到10分的幾個答案我們可以清楚的看到,小小的strcpy竟然暗藏著這麼多玄機,真不是蓋的!需要多麼紮實的基本功才能寫一個完美的strcpy啊!