news 2026/2/25 7:27:46

C语言内存函数介绍和模拟实现

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
C语言内存函数介绍和模拟实现

当我们超越字符串的边界,进入更底层的领域,便会遇见直接操作内存的工具——mem系列函数。它们不关心数据是字符、数字还是结构体,仅以原始的字节视角高效处理内存块。无论是大块数据的快速复制(memcpy)、重叠区域的稳妥搬运(memmove),还是内存的批量初始化(memset)与精确比较(memcmp),这些函数都是系统编程、算法实现和性能优化的基石。掌握它们,意味着你真正开始以计算机的方式“思考”数据在内存中的流动与组织。

目录

1.memcpy函数使用和模拟实现

2.memmove函数使用和模拟实现

3.memset函数使用和模拟实现

4.memcmp函数使用和模拟实现


1.memcpy函数使用和模拟实现

函数原型:

void* memcpy(void* destination, const void* source, size_t num);
  • destination:目标内存起始地址(接收拷贝数据)

  • source:源内存起始地址(提供拷贝数据)

  • num:要拷贝的字节数(核心:不是元素数!)

作用:将一段内存的二进制数据 “原样拷贝” 到另一段内存

函数遇到'\0' 的时候并不会停下来,不处理内存重叠,重叠会导致数据错误,拷贝字符串时,不会自动添加'\0'

int main() { int arr1[] = { 1,2,3,4,5,6,7,8,9,10 }; int arr2[10] = { 0 }; memcpy(arr2, arr1, 20); int i = 0; for (i = 0; i < 10; i++) { printf("%d ", arr2[i]); } //1 2 3 4 5 0 0 0 0 0 return 0; }

模拟实现:

void* my_memcpy(void* destination, const void* source, size_t num) { assert(destination); assert(source); void* ret = destination; while (num) { *(char*)destination = *(char*)source; destination = (char*)destination + 1; source = (char*)source + 1; num--; } return ret; } int main() { int arr1[20] = { 0 }; int arr2[] = { 1,2,3,4,5,6,7,8,9,10 }; my_memcpy(arr1, arr2, 12); int i = 0; for (i = 0; i < 20; i++) { printf("%d ", arr1[i]); } return 0; }

2.memmove函数使用和模拟实现

函数原型:

void* memmove(void* destination, const void* source, size_t num);
  • destination:目标内存起始地址(接收拷贝数据)

  • source:源内存起始地址(提供拷贝数据)

  • num:要拷贝的字节数(核心:不是元素数!)

作用:支持内存重叠的通用内存拷贝函数,核心是在memcpy基础上解决了源内存和目标内存重叠时拷贝数据错误的问题。

int main() { int arr1[] = { 1,2,3,4,5,6,7,8,9,10 }; memmove(arr1 + 2, arr1, 20); int i = 0; for (i = 0; i < 10; i++) { printf("%d ", arr1[i]); } //1 2 1 2 3 4 5 8 9 10 return 0; }

模拟实现:

void* my_memmove(void* destination, const void* source, size_t num) { assert(destination); assert(source); void* pcur = destination; char* p = (char*)destination; char* q = (char*)source; if (p < q) { while (num--) { *p++ = *q++; } } else { p += num - 1; q += num - 1; while (num--) { *p-- = *q--; } } return pcur; } int main() { int arr[10] = { 1,2,3,4,5,6,7,8,9,10 }; my_memmove(arr+2, arr, 20); int i = 0; for (i = 0; i < 10; i++) { printf("%d ", arr[i]); } return 0; }

3.memset函数使用和模拟实现

函数原型:

void* memset(void* ptr, int value, size_t num);

作用:memset 逐字节初始化内存,将指定内存区域的每一个字节统一设置为同一个单字节值。

int main() { char str[] = "hello world"; memset(str, 'x', 6); printf(str); //xxxxxxworld return 0; }

模拟实现:

void* my_memset(void* ptr, int value, size_t num) { assert(ptr != NULL); void* start_ptr = ptr; char fill_byte = (char)value; char* byte_ptr = (char*)ptr; while (num--) { *byte_ptr = fill_byte; byte_ptr++; } return start_ptr; } int main() { char str[20]; my_memset(str, 'a', 5); str[5] = '\0'; printf("char数组初始化:%s\n", str); // aaaaa return 0; }

4.memcmp函数使用和模拟实现

函数原型:

int memcmp(const void* ptr1, const void* ptr2, size_t num);

作用:从内存起始地址开始,按字节的二进制值逐位对比,直到发现首个不同字节完成num字节的比较;若ptr1对应字节的二进制值小于ptr2,返回负数;若两段内存前num个字节完全相等,返回 0;若ptr1对应字节大于ptr2,返回正数(返回值为首个不同字节的差值)

int main() { char buffer1[] = "DWgaOtP12df0"; char buffer2[] = "DWGAOTP12DF0"; int n; n = memcmp(buffer1, buffer2, sizeof(buffer1)); if (n > 0) printf("'%s' is greater than '%s'.\n", buffer1, buffer2); else if (n < 0) printf("'%s' is less than '%s'.\n", buffer1, buffer2); else printf("'%s' is the same as '%s'.\n", buffer1, buffer2); //'DWgaOtP12df0' is greater than 'DWGAOTP12DF0'. return 0; }

模拟实现:

int my_memcmp(const void* ptr1, const void* ptr2, size_t num) { assert(ptr1 != NULL && ptr2 != NULL); const char* p1 = (const char*)ptr1; const char* p2 = (const char*)ptr2; while (num--) { if (*p1 != *p2) { return (char)*p1 - (char)*p2; } p1++; p2++; } return 0; } int main() { char str1[] = "abcde"; char str2[] = "abxde"; int ret1 = my_memcmp(str1, str2, 5); printf("%d\n", ret1); // 'c'(99)-'x'(120)=-21 int arr1[] = { 1, 2, 3 }; // 小端存储:01 00 00 00 | 02 00 00 00 | 03 00 00 00 int arr2[] = { 1, 4, 3 }; // 小端存储:01 00 00 00 | 04 00 00 00 | 03 00 00 00 int ret2 = my_memcmp(arr1, arr2, sizeof(arr1)); printf("%d\n", ret2); // 02-04=-2 return 0; }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/2/25 5:57:30

5分钟极速搭建抖音直播弹幕实时监控系统

5分钟极速搭建抖音直播弹幕实时监控系统 【免费下载链接】DouyinLiveWebFetcher 抖音直播间网页版的弹幕数据抓取&#xff08;2024最新版本&#xff09; 项目地址: https://gitcode.com/gh_mirrors/do/DouyinLiveWebFetcher 还在手动记录直播间互动数据吗&#xff1f;想…

作者头像 李华
网站建设 2026/2/24 3:16:01

ComfyUI-Manager升级攻略:告别版本冲突的终极方案

ComfyUI-Manager升级攻略&#xff1a;告别版本冲突的终极方案 【免费下载链接】ComfyUI-Manager 项目地址: https://gitcode.com/gh_mirrors/co/ComfyUI-Manager 还在为ComfyUI-Manager升级后节点失效而苦恼吗&#xff1f;&#x1f92f; 作为ComfyUI生态中管理节点安装…

作者头像 李华
网站建设 2026/2/18 13:03:10

如何快速部署BGE-Large-zh-v1.5:中文语义理解的终极指南

BGE-Large-zh-v1.5中文文本嵌入模型正在改变中文NLP应用的开发方式。无论你是构建智能搜索系统、文档聚类工具还是问答机器人&#xff0c;这款高性能模型都能提供精准的语义理解能力。本文将带你从零开始&#xff0c;用最简单的方式完成完整部署。 【免费下载链接】bge-large-z…

作者头像 李华
网站建设 2026/2/24 13:23:38

跨平台输入法词库转换:一键迁移你的个性化输入习惯

跨平台输入法词库转换&#xff1a;一键迁移你的个性化输入习惯 【免费下载链接】imewlconverter ”深蓝词库转换“ 一款开源免费的输入法词库转换程序 项目地址: https://gitcode.com/gh_mirrors/im/imewlconverter 还在为更换输入法时无法保留多年积累的个性化词库而烦…

作者头像 李华
网站建设 2026/2/23 16:17:30

小白指南:在树莓派上动手实验framebuffer显示功能

从零开始&#xff1a;在树莓派上玩转 framebuffer&#xff0c;亲手点亮屏幕的每一像素 你有没有想过&#xff0c;当你在树莓派上运行一个图形程序时&#xff0c;那些颜色是怎么“走”到屏幕上的&#xff1f;不是靠魔法&#xff0c;也不是神秘的黑盒——背后其实是一块内存&…

作者头像 李华
网站建设 2026/2/25 1:28:08

词库转换神器:让输入法迁移变得简单高效

词库转换神器&#xff1a;让输入法迁移变得简单高效 【免费下载链接】imewlconverter ”深蓝词库转换“ 一款开源免费的输入法词库转换程序 项目地址: https://gitcode.com/gh_mirrors/im/imewlconverter 还在为更换输入法而烦恼吗&#xff1f;深蓝词库转换工具正是你需…

作者头像 李华