模拟实现库函数strcpy、strlen、strcat、strstr、strchr、strcmp、memcpy、memmove

本文主要是介绍模拟实现库函数strcpy、strlen、strcat、strstr、strchr、strcmp、memcpy、memmove,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

库函数strcpy的模拟实现:把从src地址开始且含有’\0’结束符的字符串复制到以dest开始的地址空间,返回值的类型为char*

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
char * my_strcpy(char *dest, const char *src)
{char *ret = dest;assert(ret != NULL);assert(src != NULL);while (*ret++ = *src++);return dest;
}
int main()
{char arr[] = "123456";printf("%s\n", my_strcpy(arr,"abcdef"));system("pause");return 0;
}


里面有4 点注意事项:

  • 返回类型 char *,实现链式访问。

  • const修饰的源指针src,具有常属性,不会被更改。

  • assert断言,保证健壮性、鲁棒性。

  • while循环自带指针++,优化代码。

库函数strlen的模拟实现:计算给定字符串的长度

1.创建计数变量

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
int my_strlen(const char *src)
{int count=0;assert(src != NULL);while (*(src++))count++;return count;
}
int main()
{printf("%d\n", my_strlen("abcdef"));system("pause");return 0;
}

2.使用递归

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
int my_strlen(const char *str)
{assert(str != NULL);if (*str == '\0')return 0;elsereturn 1 + my_strlen(str + 1);
}
int main()
{printf("%d\n", my_strlen("abcdef"));system("pause");return 0;
}

3 使用指针-指针的方式

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
int my_strlen(const char *str)
{const char *p = str;assert(str != NULL);while (*p != '\0'){p++;}return p - str;
}
int main()
{printf("%d\n", my_strlen("abcdef"));system("pause");return 0;
}

库函数strcat的模拟实现:字符串连接函数

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
char *my_strcat(char *dest, const char *src)
{char *ret = dest;assert(dest != NULL);assert(src != NULL);while (*dest)dest++;while (*dest++ = *src++);return ret;
}
int main()
{char arr[20] = "hello ";printf("%s\n", my_strcat(arr, "friend"));system("pause");return 0;
}

库函数strstr的模拟实现:从字符串str1中查找是否有字符串str2,如果有,从str1中的str2位置起,返回str1的指针,如果没有,返回NULL

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <string.h>
char *my_strstr(const char *dest, const char *src)
{const char *s1 = dest;const char *s2 = src;assert(dest && src != NULL);if (*s2 == '\0')return (char*)dest;while (*dest != '\0'){s1 = dest;s2 = src;while (*s1 && *s2 && *s1 == *s2){s1++;s2++;}if (*s2 == '\0'){return (char*)dest;}dest++;}return NULL;
}
int main()
{char arr1[] = "abbbbcdef";char arr2[] = "bbc";char * ret =my_strstr(arr1, arr2);if (ret != NULL)printf("%s\n", ret);elseprintf("没找到\n");system("pause");return 0;
}

库函数strchr的模拟实现:返回首次出现_Val的位置的指针,返回的地址是被查找字符串指针开始的第一个与Val相同字符的指针,如果Str中不存在Val则返回NULL

char *my_strchr(const char *p, int chr)
{const char *s = p;assert(p);while (*s != (char)chr && *s)s++;if (*s == (char)chr)return (char *)s;return NULL;
}
int main()
{char *p = "abcdef";char chr = 'c';char *ret = my_strchr(p, chr);if (ret != NULL)printf("找到了,%s\n", ret);elseprintf("没找到\n");system("pause");return 0;
}

库函数strcmp的模拟实现:比较两个字符串。设这两个字符串为str1,str2,若str1==str2,则返回零 ;若str1 < str2,则返回负数; 若str1 > str2,则返回正数。

方法1:

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <string.h>
int my_strcmp(const char *dest, const char *src)
{assert(dest&&src != NULL);while (*dest == *src && *src){dest++;src++;  }return *dest - *src;
}
int main()
{char arr1[] = "abcdef";char arr2[] = "aaaaaaa";int ret = my_strcmp(arr1, arr2);printf("ret=%d\n", ret);system("pause");return 0;
}


方法2:(库函数版)

int my_strcmp(const char *dest, const char *src)
{int ret = 0;assert(dest && src != NULL);while (!(ret = (unsigned char *)dest - (unsigned char *)src) && *dest){dest++;src++;}if (ret < 0)ret = -1;else if (ret>0)ret = 1;return ret;
}

我们发现strcpy是不能拷贝内容为0的字符串的内容,所以我们引入了memcpy

库函数memcpy的模拟实现:内存拷贝函数,从源src所指的内存地址的起始位置开始拷贝n个字节到目标dest所指的内存地址的起始位置中

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <string.h>
void *my_memcpy(void *dest, const void *src, int count)
{void *cp = dest;assert(dest && src);while (count--){*(char *)dest = *(char *)src;++(char *)dest;++(char *)src;}return cp;
}int main()
{int arr1[5] = { 1, 2, 3, 4, 5 };int arr2[5] = { 6, 7, 8, 9, 0 };int i = 0;my_memcpy(arr1, arr2, 20);for (i = 0; i < 5;i++)printf("%d ",arr1[i]);printf("\n");system("pause");return 0;
}

库函数memmove的模拟实现:用于从src拷贝count个字节到dest,如果目标区域和源区域有重叠的话,memmove能够保证源串在被覆盖之前将重叠区域的字节拷贝到目标区域中。但复制后src内容会被更改。但是当目标区域与源区域没有重叠则和memcpy函数功能相同

void *my_memmove(void *dest, const void *src, int count)
{void *cp = dest;assert(dest&&src);if (dest < src){while (count--){*(char *)dest = *(char *)src;++(char *)dest;++(char *)src;}}else{while (count--){*((char *)dest + count) = *((char *)src + count);}}return cp;
}
int main()
{int arr[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };int i = 0;int sz = sizeof(arr) / sizeof(arr[0]);my_memmove(arr, arr+2, 16);for (i = 0; i < sz; i++)printf("%d ", arr[i]);printf("\n");system("pause");return 0;
}


这篇关于模拟实现库函数strcpy、strlen、strcat、strstr、strchr、strcmp、memcpy、memmove的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

usaco 1.2 Transformations(模拟)

我的做法就是一个一个情况枚举出来 注意计算公式: ( 变换后的矩阵记为C) 顺时针旋转90°:C[i] [j]=A[n-j-1] [i] (旋转180°和270° 可以多转几个九十度来推) 对称:C[i] [n-j-1]=A[i] [j] 代码有点长 。。。 /*ID: who jayLANG: C++TASK: transform*/#include<

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略 1. 特权模式限制2. 宿主机资源隔离3. 用户和组管理4. 权限提升控制5. SELinux配置 💖The Begin💖点点关注,收藏不迷路💖 Kubernetes的PodSecurityPolicy(PSP)是一个关键的安全特性,它在Pod创建之前实施安全策略,确保P

工厂ERP管理系统实现源码(JAVA)

工厂进销存管理系统是一个集采购管理、仓库管理、生产管理和销售管理于一体的综合解决方案。该系统旨在帮助企业优化流程、提高效率、降低成本,并实时掌握各环节的运营状况。 在采购管理方面,系统能够处理采购订单、供应商管理和采购入库等流程,确保采购过程的透明和高效。仓库管理方面,实现库存的精准管理,包括入库、出库、盘点等操作,确保库存数据的准确性和实时性。 生产管理模块则涵盖了生产计划制定、物料需求计划、

C++——stack、queue的实现及deque的介绍

目录 1.stack与queue的实现 1.1stack的实现  1.2 queue的实现 2.重温vector、list、stack、queue的介绍 2.1 STL标准库中stack和queue的底层结构  3.deque的简单介绍 3.1为什么选择deque作为stack和queue的底层默认容器  3.2 STL中对stack与queue的模拟实现 ①stack模拟实现