memcpy
是 C 语言中非常重要的一个函数,用于内存块的复制 。它定义在标准头文件 <string.h>
中。
📌 函数原型
c
void *memcpy(void *dest, const void *src, size_t n);
参数说明:
参数 | 类型 | 含义 |
---|---|---|
dest |
void * |
目标内存地址(要复制到哪里) |
src |
const void * |
源内存地址(从哪里复制) |
n |
size_t |
要复制的字节数 |
返回值:
- 返回指向目标内存
dest
的指针(即第一个参数)。 - 通常可以忽略返回值,但在链式调用或需要检查时有用。
✅ 使用步骤与要点
-
包含头文件
c#include <string.h>
-
确保目标空间足够大
dest
必须有足够的内存空间容纳n
个字节,否则会导致缓冲区溢出和未定义行为。
-
避免内存重叠
- 如果源和目标内存区域有重叠 ,使用
memcpy
会导致未定义行为。 - 此时应使用
memmove
,它是安全处理重叠内存的版本。
- 如果源和目标内存区域有重叠 ,使用
💡 示例代码
示例 1:基本用法 ------ 复制数组
c
#include <stdio.h>
#include <string.h>
int main() {
int src[] = {10, 20, 30, 40, 50};
int dest[5];
// 复制 5 个整数(每个 int 通常是 4 字节,共 20 字节)
memcpy(dest, src, sizeof(src));
for (int i = 0; i < 5; i++) {
printf("%d ", dest[i]); // 输出: 10 20 30 40 50
}
return 0;
}
示例 2:复制结构体
c
#include <stdio.h>
#include <string.h>
typedef struct {
int id;
char name[20];
float score;
} Student;
int main() {
Student s1 = {1001, "Alice", 95.5f};
Student s2;
memcpy(&s2, &s1, sizeof(Student));
printf("ID: %d, Name: %s, Score: %.1f\n",
s2.id, s2.name, s2.score);
// 输出: ID: 1001, Name: Alice, Score: 95.5
return 0;
}
示例 3:复制字符串(非最佳实践)
c
#include <stdio.h>
#include <string.h>
int main() {
char src[] = "Hello, world!";
char dest[50];
memcpy(dest, src, strlen(src) + 1); // +1 包含 '\0'
printf("Copied string: %s\n", dest);
return 0;
}
⚠️ 注意:对于字符串,推荐使用
strcpy
或更安全的strncpy
/strlcpy
/snprintf
。
⚠️ 常见错误与陷阱
错误 | 说明 |
---|---|
❌ 目标空间不足 | char dest[10]; memcpy(dest, src, 100); → 写越界,崩溃或安全漏洞 |
❌ 源指针为空 | memcpy(dest, NULL, n) → 崩溃 |
❌ 目标指针为空 | memcpy(NULL, src, n) → 崩溃 |
❌ 内存重叠 | memcpy(buf+1, buf, 10); → 未定义行为!应改用 memmove |
✅ 正确做法示例(使用 memmove
处理重叠):
c
char buf[] = "abcdef";
memmove(buf + 1, buf, 5); // 将 "abcde" 左移一位
// 结果: "aabcde"
🔍 memcpy
vs memmove
函数 | 是否支持内存重叠 | 性能 | 推荐场景 |
---|---|---|---|
memcpy |
❌ 不保证 | 更快 | 确定无重叠时使用 |
memmove |
✅ 安全支持 | 稍慢 | 可能存在重叠时使用 |
建议 :如果不确定是否重叠,优先使用
memmove
。
🛠️ 技巧:复制部分数据
c
double data[100];
double subset[10];
// 只复制前 10 个 double
memcpy(subset, data, 10 * sizeof(double));
✅ 安全使用建议
- 使用
sizeof()
计算大小,避免硬编码。 - 确保目标缓冲区已分配且足够大。
- 对于字符串操作,考虑使用更高层函数(如
strcpy
,strncpy
)。 - 避免跨线程共享未保护的内存拷贝。
- 在嵌入式或性能敏感场景中,
memcpy
是高效选择。
🧪 小结:一句话记住 memcpy
"按字节复制一块内存,不管内容是什么。"
它不关心你复制的是 int
、float
、结构体还是字符串 ------ 它只是把一串字节从一个地方搬到另一个地方。
如果你是在实现序列化、解析协议、操作二进制数据等底层任务,memcpy
是不可或缺的工具。但务必小心使用,防止内存错误!