0

0

C++ 怎么实现字符串拷贝 C++ strcpy与strncpy安全版本【C风格】

穿越時空

穿越時空

发布时间:2026-02-01 13:43:21

|

501人浏览过

|

来源于php中文网

原创

C++中不应直接使用strcpy和strncpy,因二者均不检查缓冲区大小且strncpy不保证null终止,易致缓冲区溢出或未定义行为;应优先使用std::string,必要时用snprintf或封装安全函数。

c++ 怎么实现字符串拷贝 c++ strcpy与strncpy安全版本【c风格】

为什么不能直接用 strcpystrncpy

这两个函数在 C++ 中仍可调用(来自 ),但它们不检查目标缓冲区大小,也不保证结果字符串以 '\0' 结尾——strcpy 可能越界写入,strncpy 在源串长度 ≥ 目标大小时不会补 '\0',后续用 strlenprintf 就会读越界。常见错误现象是程序崩溃、输出乱码或堆破坏。

典型误用:

char dst[5];
strcpy(dst, "hello"); // 写入 6 字节(含 '\0'),dst 只有 5 字节 → 缓冲区溢出
char dst[5];
strncpy(dst, "hello", sizeof(dst)); // 拷贝前 5 字节 "hello",不加 '\0' → dst 不是合法 C 字符串

strlcpy 是更安全的替代,但不是标准 C/C++ 函数

strlcpy 在 BSD 系统和部分 Linux 发行版(如 glibc 2.39+)中可用,它明确要求目标缓冲区大小,并始终以 '\0' 结尾(除非 size == 0)。但它**不是 ISO C++ 标准函数**,跨平台项目中需谨慎:

  • Windows 默认不提供,需自行实现或引入兼容层
  • 旧版 glibc(如 CentOS 7)不支持
  • 参数顺序是 strlcpy(dst, src, size),与 strncpy 一致,但语义更清晰

简易自实现(仅作示意,生产环境建议用 std::string 或已验证的库):

立即学习C++免费学习笔记(深入)”;

宣小二
宣小二

宣小二:媒体发稿平台,自媒体发稿平台,短视频矩阵发布平台,基于AI驱动的企业自助式投放平台。

下载
size_t strlcpy(char* dst, const char* src, size_t size) {
    if (size == 0) return strlen(src);
    size_t len = strlen(src);
    size_t copy_len = (len < size - 1) ? len : size - 1;
    memcpy(dst, src, copy_len);
    dst[copy_len] = '\0';
    return len;
}

真正推荐的 C 风格安全做法:用 snprintf

snprintf 是 ISO C99 起就标准化的函数,C++11 及以后可直接用(包含 ),它天然带长度限制且保证末尾 '\0'

  • 总是返回「若缓冲区足够大时本应写入的字符数」,可用于判断是否截断
  • 即使 size == 0 也安全(只计算长度,不写内存)
  • 支持格式化,但单纯拷贝字符串时用 "%s" 即可

示例:

char dst[10];
int ret = snprintf(dst, sizeof(dst), "%s", "hello world");
// ret == 11 → 实际写入了 "hello wor" + '\0'(共 10 字节),说明被截断
if (ret >= (int)sizeof(dst)) {
    // 拷贝不完整,需处理
}

C++ 项目里其实不该用这些 C 风格函数

除非你明确在写系统层、嵌入式驱动或与 C ABI 交互的接口,否则在普通 C++ 代码中坚持用 strcpy 类函数,等于主动放弃类型安全和 RAII。真正该做的:

  • std::string 存储和传递文本,拷贝就是 std::string s2 = s1;,无缓冲区管理负担
  • 需要 C 兼容接口时,用 s.c_str() 获取只读指针;必须写入 C 缓冲区时,先确认大小,再用 snprintfstd::copy_n + 手动置 '\0'
  • 若必须用 C 风格字符串操作,至少封装一层检查:比如写个 safe_strcpy(dst, sizeof(dst), src) 并在 debug 版本做断言

最常被忽略的一点:很多开发者以为「用了 strncpy 就安全了」,却忘了它不自动补 '\0',而这个缺失的空字符,在后续任意一次 strcmpstrcat 中都可能引发未定义行为。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

503

2023.08.02

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

237

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

499

2024.03.01

printf用法大全
printf用法大全

php中文网为大家提供printf用法大全,以及其他printf函数的相关文章、相关下载资源以及各种相关课程,供大家免费下载体验。

75

2023.06.20

fprintf和printf的区别
fprintf和printf的区别

fprintf和printf的区别在于输出的目标不同,printf输出到标准输出流,而fprintf输出到指定的文件流。根据需要选择合适的函数来进行输出操作。更多关于fprintf和printf的相关文章详情请看本专题下面的文章。php中文网欢迎大家前来学习。

286

2023.11.28

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

361

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

212

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1503

2023.10.24

go语言 注释编码
go语言 注释编码

本专题整合了go语言注释、注释规范等等内容,阅读专题下面的文章了解更多详细内容。

30

2026.01.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 8.2万人学习

Git 教程
Git 教程

共21课时 | 3.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号