您的位置:首页 > 其它

库函数strcpy/strlen的工作方式

2015-10-27 17:07 176 查看
库函数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啊!
读者看了不同分值的strcpy版本,应该也可以写出一个10分的strlen函数了,完美的版本为:

int strlen( const char *str ) //输入参数const

{

 assert( strt != NULL ); //断言字符串地址非0

 int len;

 while( (*str++) != '\0' )

 {

  len++;

 }

 return len;

}



strcpy得到了很多面试人员的偏爱,其很大的体现出了C语言基本功.

下面是strcpy原函数

char *strcpy(char *strDest, const char *strSrc);

{

assert((strDest!=NULL)&& (strSrc !=NULL));

char *address = strDest;

while( (*strDest++ = * strSrc++) != ‘\0’ )

NULL ;

return address ;

}

有几个问题值得思考并注意:

1.用指针作为形参,在程序开始前应判断指针变量的合法性,利用断言assert

2.只作为输入参数,为了防止被编程人员恶意修改,应加入常量声明const

3.函数返回类型为,char *,其目的是为了方便实现链表表达式.在调用函数的过程中已经实现了把strSrc的内容复制到strDest

4.应注意对原始位置的保留.

5.该函数调用完成时是遇到'\0'结束符

下面是几个找错题:

试题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会导致数组越界;从第5条可以看出,函数调用完成时已经将'\0'一同赋给了目的地址.

试题2,问题很严重,对string[10]数组的赋值本身没有错误,但很明显程序期望对string按字符串操作,这就需要额外的赋给一个'\0'结束符;另外如果没有'\0'结束标志,调用strcpy将不会结束,造成程序的不确定性,很危险.

试题3,一定要注意strlen(pstr),它和strcpy不同,遇到'\0'时结束,但并不计算\0'在内.所以应改为if( strlen( str1 ) <=10 ),另外要注意字符串首地址,数组,各种类型的指针作为形参,都退化为指针.即

void test3(char* str1)

{

sizeof(str1)只和编译器和CPU类型有关,32位CPU为4
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: