【C/C++】字符串拷贝方法

以下是 C/C++ 中常见字符串拷贝方法的详细对比,涵盖安全性、性能、使用场景和现代最佳实践:


1. strcpy (C风格,不安全)

c 复制代码
char* strcpy(char* dest, const char* src);
  • 特点

    • 不检查目标缓冲区大小,可能缓冲区溢出
    • 自动添加 \0
    • 性能最高(无额外检查)。
  • 适用场景

    • 仅用于**100%确定src长度不超过dest**的情况(如硬编码短字符串)。
  • 风险示例

    c 复制代码
    char dest[5];
    strcpy(dest, "hello world"); // 崩溃!缓冲区溢出

2. strncpy (部分安全)

c 复制代码
char* strncpy(char* dest, const char* src, size_t n);
  • 特点

    • 限制最大拷贝字符数n,防止溢出。

    • 不会自动补\0 (若n <= strlen(src)),需手动处理:

      c 复制代码
      dest[n-1] = '\0'; // 手动终止
    • 性能接近strcpy

  • 适用场景

    • 固定长度缓冲区(如嵌入式系统)。
  • 示例

    c 复制代码
    char dest[10];
    strncpy(dest, "hello world", sizeof(dest));
    dest[sizeof(dest)-1] = '\0'; // 必须手动终止

3. snprintf (推荐,安全)

c 复制代码
int snprintf(char* dest, size_t size, const char* format, ...);
  • 特点

    • 完全安全:自动限制写入长度(最多size-1字符),保证\0结尾
    • 支持格式化(如拼接字符串、数字)。
    • 性能较低(需解析格式字符串)。
  • 适用场景

    • 安全拷贝或格式化字符串。
    • 现代C代码首选。
  • 示例

    c 复制代码
    char dest[10];
    snprintf(dest, sizeof(dest), "%s", "hello world"); // 自动截断并补\0

4. memcpy (二进制拷贝)

c 复制代码
void* memcpy(void* dest, const void* src, size_t n);
  • 特点

    • 直接内存拷贝,不检查\0,适合任意二进制数据。
    • 高性能(比strcpy更底层)。
    • 手动管理长度和\0
  • 适用场景

    • 已知长度的字符串或二进制数据。
    • 高性能需求(如大规模数据拷贝)。
  • 示例

    c 复制代码
    char dest[10];
    memcpy(dest, "hello", 6); // 包含\0

5. std::string (C++推荐)

cpp 复制代码
std::string dest = src;
  • 特点

    • 绝对安全:自动管理内存,无需关心缓冲区大小。
    • 支持动态扩容、拼接、查找等操作。
    • 性能略低于C风格(因堆分配)。
  • 适用场景

    • 所有C++字符串操作场景。
    • 需要高可读性和安全性的代码。
  • 示例

    cpp 复制代码
    std::string src = "hello";
    std::string dest = src; // 自动深拷贝

对比总结

方法 安全性 自动补\0 性能 适用场景
strcpy ❌ 不安全 ✅ 是 ⭐⭐⭐ 已知安全的短字符串
strncpy ⚠️ 部分安全 ❌ 需手动 ⭐⭐ 固定长度缓冲区
snprintf ✅ 安全 ✅ 是 安全拷贝或格式化字符串
memcpy ⚠️ 需手动 ❌ 需手动 ⭐⭐⭐ 已知长度的二进制数据/字符串
std::string ✅ 安全 ✅ 是 ⭐⭐ 所有C++字符串操作

最佳实践建议

  1. C++代码 :优先使用 std::string,避免手动管理内存。
  2. C代码安全拷贝 :用 snprintf(自动截断)或 strncpy + 手动补\0
  3. 高性能场景memcpy(需确保长度正确)。
  4. 绝对避免 :原生strcpy,除非能严格保证安全。

代码示例(安全拷贝)

cpp 复制代码
// C++ (推荐)
std::string src = "hello";
std::string dest = src;

// C (安全方案)
char dest[10];
snprintf(dest, sizeof(dest), "%s", "hello world"); // 自动处理

// C (高性能场景)
char dest[10];
memcpy(dest, "hello", 6); // 明确包含\0

特殊场景处理

  • Windows平台 :可用 strcpy_s(C11扩展)。
  • 动态缓冲区 :C用malloc+strcpy,C++用std::string
相关推荐
Kay_Liang4 小时前
探究排序算法的奥秘(下):快速排序、归并排序、堆排序
java·数据结构·c++·python·算法·排序算法
泪光29295 小时前
枚举法——C++算法【泪光2929】
c++
李匠20247 小时前
C++负载均衡远程调用学习之基础TCP服务
c++·学习
Berserker_D7 小时前
【C/C++】头文件防卫式宏
c语言·开发语言·c++
绒绒毛毛雨7 小时前
将infinigen功能集成到UE5--在ue里面写插件(python和c++)
c++·python·ue5
海码0077 小时前
【Hot 100】 148. 排序链表
数据结构·c++·链表·排序算法·hot100
余弦的倒数8 小时前
C++的vector中emplace_back() 与 push_back() 的区别
开发语言·c++
鱼糕权八郎 -8 小时前
LeetCode392_判断子序列
c++·leetcode
到底怎么取名字不会重复8 小时前
Day16(贪心算法)——LeetCode45.跳跃游戏II&763.划分字母区间
c++·算法·leetcode·游戏·贪心算法