一、完整的 strcpy 实现代码
cpp
// 实现strcpy:将strSrc拷贝到strDest,返回strDest(兼容库函数原型)
char* strcpy(char* strDest, const char* strSrc)
{
// 1. 健壮性校验:空指针直接返回(或根据需求抛异常,C风格常用返回NULL)
if (strDest == nullptr || strSrc == nullptr)
{
return nullptr;
}
// 2. 保存目标字符串起始地址(用于最后返回)
char* pDest = strDest;
// 3. 逐字节拷贝:直到遇到源字符串的结束符'\0'
while (*strSrc != '\0')
{
*strDest = *strSrc; // 拷贝当前字符
strDest++; // 目标指针后移
strSrc++; // 源指针后移
}
// 4. 关键:给目标字符串末尾补'\0'(字符串的核心标志)
*strDest = '\0';
// 5. 返回目标字符串起始地址(兼容库函数的返回值设计)
return pDest;
}
// 测试用例(验证功能)
#include <iostream>
int main()
{
// 测试1:正常拷贝
char dest1[20] = {0}; // 初始化避免脏数据
const char* src1 = "hello strcpy";
strcpy(dest1, src1);
std::cout << "正常拷贝结果:" << dest1 << std::endl; // 输出:hello strcpy
// 测试2:源字符串为空
char dest2[20] = {0};
const char* src2 = "";
strcpy(dest2, src2);
std::cout << "空字符串拷贝结果:" << (dest2[0] == '\0' ? "空字符串" : "非空") << std::endl; // 输出:空字符串
// 测试3:空指针(验证健壮性)
char* dest3 = nullptr;
const char* src3 = "test null";
char* res = strcpy(dest3, src3);
std::cout << "空指针拷贝结果:" << (res == nullptr ? "返回NULL" : "非NULL") << std::endl; // 输出:返回NULL
return 0;
}
二、核心逻辑拆解
-
空指针校验 这是工业级代码的必备逻辑 ------ 如果
strDest或strSrc是NULL,直接返回NULL,避免程序崩溃(新手常忽略这一步,只写拷贝逻辑)。也可以用assert(strDest && strSrc);(断言),调试阶段快速发现空指针问题,但断言在 Release 模式下会失效,所以空指针判断更通用。 -
保存起始地址 因为拷贝过程中
strDest会不断后移,需要先用pDest记住原始起始位置,最后返回(和库函数strcpy的行为一致)。 -
逐字节拷贝
*strSrc表示取strSrc指针指向的当前字符;- 循环条件
*strSrc != '\0':字符串的结束标志是'\0',只要没到末尾就继续拷贝; - 每次拷贝后,两个指针都向后移动一位(
strDest++/strSrc++)。
-
末尾补 '\0' 这是最容易踩坑的点!源字符串的
'\0'不会被循环拷贝(循环到'\0'就终止了),必须手动给目标字符串加'\0',否则目标字符串会变成 "乱码"(因为找不到结束符)。