当前位置: 首页 > news >正文

C语言菜鸟入门·浅析strdup和strcpy的区别

目录

1.  strdup

2.  strcpy

3.  区别


1.  strdup

        strdup 是 C 语言中一个常用的非标准库函数(主要出现在 POSIX 系统,如 Linux/Unix),用于动态复制字符串。其功能是分配足够的内存来复制给定的字符串,并返回指向新字符串的指针。

        也就是说 strdup 复制完的数据有自己的存放地址,和原数据没有关联。

//函数原型#include <string.h> // 头文件char *strdup(const char *s);

        strdup函数复制一个字符串,使用完后,要使用free函数释放内存,strdup函数的参数不能为NULL,一旦为NULL,就会报段错误,示例代码:

#include <stdio.h>
#include <string.h>
#include <stdlib.h> // 包含 free()int main() {const char *src = "Hello, strdup!";// 复制字符串char *copy = strdup(src);if (copy == NULL) // 检查是否成功{ perror("strdup failed");return 1;}printf("Original: %s\n", src);printf("Copy:     %s\n", copy);free(copy); // 必须释放内存!return 0;
}

在使用 strdup 需要注意,其不是 C 标准库函数(C89/C99/C11/C17 均未收录),但:

  • 大多数编译器(如 GCC、Clang)支持。
  • Windows 下需定义 _POSIX_C_SOURCE 或使用 _strdup(需 <string.h>)。

        当然如果环境不支持,你也可以自行实现:

char *my_strdup(const char *s) {if (s == NULL) return NULL;size_t len = strlen(s) + 1;   // +1 用于结尾的 '\0'char *copy = (char*)malloc(len);if (copy != NULL) {memcpy(copy, s, len);     // 复制内容(包括 '\0')}return copy;
}

        详细:

#include <stdio.h>
#include <string.h>
#include <stdlib.h> // 包含 free()char *my_strdup(const char *s) {if (s == NULL) return NULL;size_t len = strlen(s) + 1;   // +1 用于结尾的 '\0'char *copy = (char*)malloc(len);if (copy != NULL) {memcpy(copy, s, len);     // 复制内容(包括 '\0')}return copy;
}int main() {const char *src = "Hello, strdup!";// 复制字符串char *copy = my_strdup(src);if (copy == NULL) // 检查是否成功{ perror("strdup failed");return 1;}printf("Original: %s\n", src);printf("Copy:     %s\n", copy);free(copy); // 必须释放内存!return 0;
}

2.  strcpy

        strcpy 是 C 语言标准库中的字符串复制函数,用于将一个字符串(包括结束符 \0)复制到另一个内存位置。与 strdup 不同,strcpy 不会自动分配内存,调用者必须确保目标缓冲区足够大。

//函数原型#include <string.h> // 必需头文件char *strcpy(char *dest, const char *src);

        示例代码:

#include <stdio.h>
#include <string.h>int main() {char src[] = "Hello, strcpy!";char dest[20]; // 目标缓冲区必须足够大!strcpy(dest, src); // 复制字符串printf("Source: %s\n", src);printf("Destination: %s\n", dest);return 0;
}

        不过在使用过程中需要注意,如果复制的字符串超过定义的内存大小,会出现缓冲区溢出的风险,例如:

char dest[5];
strcpy(dest, "This string is too long!"); // 溢出!导致未定义行为

        我们可以使用 strncpy 进行限制最大复制长度:

char dest[10];
strncpy(dest, "Hello world", sizeof(dest)-1); // 限制复制长度
dest[sizeof(dest)-1] = '\0'; // 确保字符串终止

        其中 strncpy 的函数原型:

#include <string.h>char *strncpy(char *dest, const char *src, size_t n);//dest:目标缓冲区(需预先分配内存)
//src:源字符串
//n:最大复制的字符数(包括 '\0')

        示例代码:

#include <stdio.h>
#include <string.h>int main() {const char *src = "This is a long string that may be truncated";char dest[20]; // 目标缓冲区较小// 安全复制(保留空间给\0)strncpy(dest, src, sizeof(dest) - 1);dest[sizeof(dest) - 1] = '\0'; // 手动终止printf("Source: %s\n", src);printf("Destination: %s\n", dest); // 输出截断后的字符串return 0;
}

   3.  区别

特性strcpystrdup
内存管理目标缓冲区由调用者预先分配内部动态分配内存(malloc
安全性高风险(不检查长度)需检查返回值(可能内存不足)
释放内存无需额外释放必须显式 free()
标准性ANSI C 标准函数POSIX 扩展(非 C 标准)
使用场景已知目标缓冲区大小动态创建字符串副本

4.  拓展

4.1  free

        free 是 C 语言中用于释放动态分配内存的核心函数。它与内存分配函数(malloc, calloc, realloc, strdup)配对使用,防止内存泄漏。

        其需遵守的规则:

操作✅ 正确做法❌ 危险操作
释放次数每个分配只释放一次重复释放:free(p); free(p); → 崩溃
释放后访问释放后立即置 NULLfree(p); printf("%s", p); → 未定义行为
内存所有权谁分配谁释放跨模块释放:模块A分配 → 模块B释放 → 易出错
部分释放必须释放整个内存块尝试释放分配块的中间位置 → 崩溃
静态内存只用于堆内存free(&local_var) → 灾难性错误

        举个例子:

#include <stdlib.h>int main() {// 1. 动态分配内存char *buffer = malloc(100 * sizeof(char));if (buffer == NULL) {// 处理分配失败return 1;}// 2. 使用内存...strcpy(buffer, "Dynamic memory example");// 3. 释放内存free(buffer);buffer = NULL; // 消除野指针return 0;
}

C语言_时光の尘的博客-CSDN博客

http://www.dtcms.com/a/287553.html

相关文章:

  • 1.初始化
  • 【电脑】声卡的基础知识
  • CTF misc之数字取证
  • 我做的基础服务项目,是如何实现 API 安全与限流的(短信、邮件、文件上传、钉钉通知)
  • lazyvim配置
  • 教育科技产品设计:从公司背景到 MVP 方案的落地思路
  • laravel RedisException: Connection refused优雅草PMS项目管理系统报错解决-以及Redis 详细指南-优雅草卓伊凡
  • 传统行业和AIGC的结合及应用
  • Spring AI 项目实战(十八):Spring Boot + AI + Vue3 + OSS + DashScope 实现高效语音识别系统(附完整源码)
  • PyQt5—Qt QDialog 学习笔记
  • 【RK3576】【Android14】SDK源码编译
  • 【RK3576】【Android14】UART开发调试
  • JavaScript基础语法和简单数据结构
  • 【小沐学GIS】基于Rust绘制三维数字地球Earth(Rust、OpenGL、GIS)
  • RPC(Remote Procedure Call,远程过程调用)介绍
  • MySQL数据丢失救援办法
  • 下一场范式革命:Transformer架构≠最终解法
  • 《全栈博客系统的技术肌理:从接口构建到体验升维的实践路径》
  • 美国VPS服务器Linux内核参数调优的实践与验证
  • 第二次总结(xss、js原型链)
  • 【2025最新】使用neo4j实现GraphRAG所需的向量检索
  • OAIF:基于在线 AI 反馈的语言模型直接对齐
  • [MarkdownGithub] 使用块引用高亮显示“注意“和“警告“和其他注意方式的选项
  • Django母婴商城项目实践(九)- 商品列表页模块
  • vue2 面试题及详细答案150道(121 - 130)
  • [Python] -实用技巧10- 时间处理:datetime 和 time 模块入门
  • React 的 `cache()` 函数
  • 使用 Gunicorn 部署 Django 项目
  • XSS相关理解
  • Ubuntu20.04 samba配置