我们从MSDN中找到strcpy函数原型:
char *strcpy( char *strDestination, const char *strSource);
1:不调用库函数,实现strcpy函数
2:解释strcpy代码中的各个参数意义
第一个问题我们就直接看代码吧
char *strcpy( char *strDestination, const char *strSource)
{
assert(strDestination!=NULL&&strSource!=NULL);
char *str=strDestination;
while((*strDestination++=*strDestination++)!='\0');
return str;
}
实现代码是非常简单的,一个字节一个字节的拷贝即可,当碰到0就退出即可。
针对第二个问题我们提出以下几点:
1:为什么使用const?
大家可以看出const修饰的是char类型,我们是拷贝源串,所以要防止在拷贝函数中修改了源字符串
2:空指针检查
源指针和目的指针都有可能会出现空指针的情况,所以应该对其进行检查,如果不检查指针的有效性,说明答题者不注重代码的健壮性。
3:为什么定义一个str指针?
因为目的指针strDestination已经在进行移动了,我们需要返回字符串首地址,所以用辅助指针str表明首指针。
4:检查指针的有效性时为什么不使用assert(!strDestination && !srcstrSource);
因为char *转换为bool即是类型隐式转换,这种功能虽然灵活,但更多的是导致出错概率增大和维护成本升高
5:检查指针的有效性时为什么不使用assert(strDestination != 0 && strSource != 0);,
如果这样写说明你不知道使用常量的好处。直接使用字面常量(如本例中的0)会减少程序的可维护性。0虽然简单,但程序中可能出现很多处对指针的检查,万一出现笔误,编译器不能发现,生成的程序内含逻辑错误,很难排除。而使用NULL代替0,如果出现拼写错误,编译器就会检查出来。
6:为什么要返回char *类型?
当然也可以返回void *类型,但是返回strDestination的原始值使函数能够支持链式表达式,增加了函数的“附加值”。同样功能的函数,如果能合理地提高的可用性,自然就更加理想。
链式表达式的形式如:
int iLength=strlen(strcpy(strA,strB));
又如:
char * strA=strcpy(new char[10],strB);