跳转到内容
主菜单
主菜单
移至侧栏
隐藏
导航
首页
最近更改
随机页面
MediaWiki帮助
代码酷
搜索
搜索
中文(中国大陆)
外观
创建账号
登录
个人工具
创建账号
登录
未登录编辑者的页面
了解详情
贡献
讨论
编辑“︁
C 语言安全字符串函数
”︁(章节)
页面
讨论
大陆简体
阅读
编辑
编辑源代码
查看历史
工具
工具
移至侧栏
隐藏
操作
阅读
编辑
编辑源代码
查看历史
常规
链入页面
相关更改
特殊页面
页面信息
外观
移至侧栏
隐藏
您的更改会在有权核准的用户核准后向读者展示。
警告:
您没有登录。如果您进行任何编辑,您的IP地址会公开展示。如果您
登录
或
创建账号
,您的编辑会以您的用户名署名,此外还有其他益处。
反垃圾检查。
不要
加入这个!
= C语言安全字符串函数 = == 介绍 == 在C语言中,字符串是以空字符('\0')结尾的字符数组。传统的字符串操作函数(如`strcpy`、`strcat`、`gets`等)容易导致缓冲区溢出,这是许多安全漏洞的根源。为了解决这些问题,C11标准引入了'''安全字符串函数'''(以`_s`后缀标识),这些函数要求明确指定目标缓冲区的大小,从而防止缓冲区溢出。 安全字符串函数的主要特点包括: * 显式指定缓冲区大小 * 在溢出时返回错误代码而非继续执行 * 可选的目标指针NULL检查 == 传统函数的风险 == 传统字符串函数不检查目标缓冲区大小,可能导致严重的安全问题: <syntaxhighlight lang="c"> #include <stdio.h> #include <string.h> int main() { char buffer[5]; strcpy(buffer, "This string is too long"); // 缓冲区溢出! printf("%s\n", buffer); return 0; } </syntaxhighlight> 输出可能是不可预测的,可能导致程序崩溃或安全漏洞。 == 安全字符串函数列表 == 以下是常用的安全字符串函数及其传统对应版本: {| class="wikitable" ! 传统函数 !! 安全版本 !! 描述 |- | `strcpy` | `strcpy_s` | 安全字符串复制 |- | `strcat` | `strcat_s` | 安全字符串连接 |- | `strtok` | `strtok_s` | 安全字符串分割 |- | `gets` | `gets_s` | 安全输入读取 |- | `sprintf` | `sprintf_s` | 安全格式化输出 |} == 安全函数使用示例 == === strcpy_s === <syntaxhighlight lang="c"> #include <stdio.h> #include <string.h> int main() { char dest[10]; errno_t err = strcpy_s(dest, sizeof(dest), "Hello"); if (err == 0) { printf("成功复制: %s\n", dest); } else { printf("错误: 缓冲区太小或无效参数\n"); } return 0; } </syntaxhighlight> 输出: <pre> 成功复制: Hello </pre> 如果尝试复制过长的字符串: <syntaxhighlight lang="c"> errno_t err = strcpy_s(dest, sizeof(dest), "This string is too long"); </syntaxhighlight> 输出: <pre> 错误: 缓冲区太小或无效参数 </pre> === strcat_s === <syntaxhighlight lang="c"> #include <stdio.h> #include <string.h> int main() { char dest[20] = "Hello"; errno_t err = strcat_s(dest, sizeof(dest), " World!"); if (err == 0) { printf("连接结果: %s\n", dest); } else { printf("错误: 缓冲区太小或无效参数\n"); } return 0; } </syntaxhighlight> 输出: <pre> 连接结果: Hello World! </pre> == 边界检查机制 == 安全字符串函数通过以下方式防止缓冲区溢出: 1. 检查目标指针是否为NULL 2. 比较源字符串长度与目标缓冲区大小 3. 在可能溢出时返回错误代码而非继续操作 <mermaid> graph TD A[调用安全函数] --> B{检查参数有效性} B -->|无效| C[返回错误代码] B -->|有效| D{缓冲区大小足够?} D -->|否| C D -->|是| E[执行操作] E --> F[返回成功] </mermaid> == 错误处理 == 安全函数通常返回`errno_t`类型(实际上是`int`)的错误代码。常见错误包括: * 0:成功 * EINVAL:无效参数 * ERANGE:缓冲区太小 == 实际应用案例 == 考虑一个用户登录系统,需要安全地处理用户名输入: <syntaxhighlight lang="c"> #include <stdio.h> #include <string.h> #define MAX_USERNAME_LEN 32 int authenticate() { char username[MAX_USERNAME_LEN + 1]; // +1 for null terminator printf("请输入用户名: "); if (gets_s(username, sizeof(username)) == NULL) { printf("输入错误或太长\n"); return 0; } // 安全比较 if (strcmp(username, "admin") == 0) { printf("认证成功\n"); return 1; } printf("无效用户名\n"); return 0; } </syntaxhighlight> == 数学表示 == 安全字符串函数可以形式化表示为: <math> f_{safe}(dst, size, src) = \begin{cases} dst \leftarrow src & \text{if } len(src) < size \\ \text{error} & \text{otherwise} \end{cases} </math> 其中: * <math>dst</math> 是目标缓冲区 * <math>size</math> 是目标缓冲区大小 * <math>src</math> 是源字符串 * <math>len(src)</math> 是源字符串长度(不包括空字符) == 兼容性考虑 == 1. 安全函数是C11标准的一部分,需要支持C11的编译器 2. 在Microsoft Visual Studio中,许多安全函数作为扩展提供 3. 在其他平台可能需要定义`__STDC_WANT_LIB_EXT1__`宏为1 == 最佳实践 == 1. 总是优先使用安全版本函数 2. 检查所有安全函数的返回值 3. 为缓冲区分配足够空间(包括空字符) 4. 考虑使用更高级的字符串库(如GLib)处理复杂操作 == 总结 == C语言安全字符串函数提供了更安全的字符串操作方式,能有效防止缓冲区溢出漏洞。虽然需要更多参数和错误检查,但显著提高了代码的安全性。初学者应从项目开始就养成使用安全函数的习惯,而经验丰富的开发者应审查现有代码,将传统函数替换为安全版本。 [[Category:编程语言]] [[Category:C]] [[Category:C 语言字符串]]
摘要:
请注意,所有对代码酷的贡献均被视为依照知识共享署名-非商业性使用-相同方式共享发表(详情请见
代码酷:著作权
)。如果您不希望您的文字作品被随意编辑和分发传播,请不要在此提交。
您同时也向我们承诺,您提交的内容为您自己所创作,或是复制自公共领域或类似自由来源。
未经许可,请勿提交受著作权保护的作品!
取消
编辑帮助
(在新窗口中打开)