C++|动手实现strcpy、memcpy、memmove

2024-05-26 13:28

本文主要是介绍C++|动手实现strcpy、memcpy、memmove,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

strcpy

手写一个字符串拷贝函数:

char* my_strcpy(char* dest, const char* src) {char* dest_ptr = dest;while (*src != '\0') *dest_ptr++ = *src++;*dest_ptr = '\0';return dest;
}
  1. 确定好函数原型,函数返回目标字符串 dest 的地址,复制原字符串const char *src到目标字符串char* dest
  2. 保存目标地址的起始位置:保存其实位置,因为我们待会儿还得把他返回呢。
  3. 逐字符复制:有个小技巧*dest_ptr = *src++;
  4. 添加空字符结尾:千万别忘
  5. 返回目的地址:return dest

memcpy

如果源内存区域和目的内存区域有重叠,memcpy 的行为是未定义的。在这种情况下,应该使用 memmove 函数,它能够正确处理重叠的内存区域。

memcpy 的使用场景一般在初始化数组、复制结构体或对象等。

void* my_memcpy(void* dest, const void* src, size_t count) {if (dest == NULL || src == NULL) return NULL;char* d = (char*)dest;char* s = (char*)src;while (count--) {*d++ = *s++;}return dest;
}
  1. 函数原型:需要注意的是,内存区域可能是任何类型
  2. 空哦指针检查
  3. 类型转换,都转换为char*类型,以便逐字节操作内存
  4. 逐字节复制
  5. 返回目标地址

memmove

memmove主要就是为了解决内存重叠现象。在内存重叠时我们应该使用memmove,因为它从后向前拷贝,可以避免任何src和dest的重叠情况下的内存数据拷贝。

#include <stddef.h> // for size_tvoid* my_memmove(void* dest, const void* src, size_t count) {if (dest == NULL || src == NULL) return NULL; // 检查空指针char* d = (char*)dest;const char* s = (const char*)src;if (d < s) {// 如果目标地址在源地址之前,可以从前往后复制while (count--) {*d++ = *s++;}} else {// 如果目标地址在源地址之后,从后往前复制d += count;s += count;while (count--) {*(--d) = *(--s);}}return dest;
}

memmove、memcpy源代码

/* Public domain.  */
#include <stddef.h>void *
memcpy (void *dest, const void *src, size_t len)
{char *d = dest;const char *s = src;while (len--)*d++ = *s++;return dest;
}//* Public domain.  */
#include <stddef.h>void *
memmove (void *dest, const void *src, size_t len)
{char *d = dest;const char *s = src;if (d < s)while (len--)*d++ = *s++;else{char *lasts = s + (len-1);char *lastd = d + (len-1);while (len--)*lastd-- = *lasts--;}return dest;
}

这篇关于C++|动手实现strcpy、memcpy、memmove的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1004591

相关文章

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

Python实现字典转字符串的五种方法

《Python实现字典转字符串的五种方法》本文介绍了在Python中如何将字典数据结构转换为字符串格式的多种方法,首先可以通过内置的str()函数进行简单转换;其次利用ison.dumps()函数能够... 目录1、使用json模块的dumps方法:2、使用str方法:3、使用循环和字符串拼接:4、使用字符

Linux下利用select实现串口数据读取过程

《Linux下利用select实现串口数据读取过程》文章介绍Linux中使用select、poll或epoll实现串口数据读取,通过I/O多路复用机制在数据到达时触发读取,避免持续轮询,示例代码展示设... 目录示例代码(使用select实现)代码解释总结在 linux 系统里,我们可以借助 select、

Linux挂载linux/Windows共享目录实现方式

《Linux挂载linux/Windows共享目录实现方式》:本文主要介绍Linux挂载linux/Windows共享目录实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录文件共享协议linux环境作为服务端(NFS)在服务器端安装 NFS创建要共享的目录修改 NFS 配

通过React实现页面的无限滚动效果

《通过React实现页面的无限滚动效果》今天我们来聊聊无限滚动这个现代Web开发中不可或缺的技术,无论你是刷微博、逛知乎还是看脚本,无限滚动都已经渗透到我们日常的浏览体验中,那么,如何优雅地实现它呢?... 目录1. 早期的解决方案2. 交叉观察者:IntersectionObserver2.1 Inter

Spring Gateway动态路由实现方案

《SpringGateway动态路由实现方案》本文主要介绍了SpringGateway动态路由实现方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随... 目录前沿何为路由RouteDefinitionRouteLocator工作流程动态路由实现尾巴前沿S