strcpy和strlen找错

试题 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 = '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 啊!
  (4)对 strlen 的掌握,它没有包括字符串末尾的'\0'。
  读者看了不同分值的 strcpy 版本,应该也可以写出一个 10 分
的 strlen 函数了,完美的版本为:
int strlen( const char *str ) //输入参数 const
{
assert( strt != NULL ); //断言字符串地址非 0
int len;
while( (*str++) != '\0' )
{
  len++;
}
return len;
}


你可能感兴趣的:(String,strlen,strcpy)