字符函数和字符串函数的模拟实现

news2024/11/28 2:46:33
  • 求字符串长度
    • strlen
  •  长度不受限制的字符串函数
    • strcpy
    • strcmp
    • strcat
  • 长度受限制的字符串函数
    • strnlen
    • strncmp
    • strncpy
    • strncat
  • 字符串查找
    • strstr
    • strtok
  • 错误信息报告
    • strerror
  • 内存操作函数
    • memcpy
    • memmove
    • memset
    • memcmp

  •  首先我们来看strlen
  • 字符串是以‘\0’为结束标志,strlen函数返回的是‘\0’出现的字符个数(但不包括‘\0’)
  • 我们还要注意一下就是strlen的返回值,它是无符号的(size_t)

size_t my_strlen(const char *str)
{
	assert(str);
	int count = 0;
	while (*str != '\0')
	{
		count++;
		str++;
	}

	return count;

}

int main()
{
	size_t ret = my_strlen("abcdef");
	printf("%d\n", ret);
	return 0;
}

  •  源字符串必须以‘\0’结束
  • 会将源字符串中的‘\0’拷贝到目标空间中
  • 目标空间必须足够大了,以确保能存放源字符串
char* my_strcpy(char *dest,const char *src)
{
	assert(src);
	assert(dest);
	char* ret = dest;
	while (*src != '\0')
	{
		*dest = *src;
		src++;
		dest++;
	}
	*dest = *src;

	return ret;

}

int main()
{
	char arr1[20] = "hello wolrd";
	char arr2[] = "xxxxxx";
	my_strcpy(arr1, arr2);
	printf("%s\n", arr1);
	return 0;
}
  • 首先,我们来看返回值,你是否一开始和我有一样的疑问呢?? 为什么返回值是char*的??
  • 我从《C语言和陷阱》中得到了答案,这是为了链式访问
  • strcpy传递的两个参数,第一个就是目标空间,第二个就是源字符串
  • 思路:
  • 我们要注意源字符串的‘\0’,我们是将源字符串一个一个拷贝到dest中,并且‘\0’也是要拷贝到'\0'的

 

 

  •  标准规定:
  • 第一个字符串大于第二个字符串,则返回大于0的数字
  • 第一个字符串等于第二个字符串,则返回0
  • 第一个字符串小于第二个字符串,则返回小于0的数字
  • int my_strcmp(const char *str1,const char *str2)
    {
    	assert(str1);
    	assert(str2);
    	while (*str1 == *str2)
    	{
    		if(*str1 == '\0')
    		{
    			return 0;
    		}
    		str1++;
    		str2++;
    	}
    	//if (*str1 > *str2)
    	//{
    	//	return 1;
    	//}
    	//else
    	//	return -1;
    
    	return (*str1 - *str2);
    }
    
    int main()
    {
    	int ret = my_strcmp("ab", "abc");
    	printf("%d\n", ret);
    	return 0;
    }

    思路:

  • 我们也是和之前一样的比较方法,结束标志都是看'\0‘,我们通过一个一个比较,最后c会和‘\0’比较,因为我们的c大于0,所以会返回一个小于0的数字,我们return (*str1 - *str2);  就是返回了一个大于或小于0的值,这样写就不用用if else来判断了,代码少了很多

  •  strcat,就是在目标字符串的后面加上源字符串
char* my_strcat(char *dest,const char *src)
{
	assert(dest);
	assert(src);
	char* ret = dest;
	while (*dest!='\0')
	{
		dest++;
	}
	while (*dest++ = *src++)
	{
		;
	}
	return ret;
}

int main()
{
	char arr1[20] = "hello ";
	char arr2[] = "world";
	my_strcat(arr1,arr2);
	printf("%s\n", arr1);
	return 0;
}

思路:

  • 因为我们是将arr2追加到arr1中,所以我们特别要关于arr2‘’\0'的位置,这样我们通过一个一个追加到arr1后面的位置,就可以实现了strcat的实现,最后我们为了链式访问,我们要一开始要保存一个dest一开始的地址

strncmp就是在传参后面加上了要比较的个数

int main()
{
	char arr1[] = "abcdefghijk";
	char arr2[] = "abcdef";
	int ret = strncmp(arr1, arr2, 7);
	printf("%d\n", ret);

	return 0;
}
  • 我们来看这个例子,因为我们在传参加上了要比较的个数,在这两个字符串中,前7个字符就是相同的,所以最后返回是的0

  • 所以,我们如果要用字符串的函数,
  • strnlen
  • strncmp
  • strncpy
  • strncat
  • 会安全很多,可以根据自己的需求来实现自己想达到的母的

 

 

int main()
{
	for (int i =0; i<10; i++)
	{
		printf("%d:%s\n", i, strerror(i));
	}

	return 0;
}

 这个函数就是可以给我们看错误的信息报告的

 

 

  •  我们来看内存操作的函数,它接受的参数是void* ,这是因为它可能接受浮点型、整型、 字符型,类型是不确定的,所以我们用void* 来接受
  • void* my_memcpy(void *dest,const void *src,size_t num)
    {
    	assert(dest && src);
    	void* ret = dest;
    	while (num--)
    	{
    		*(char*)dest = *(char*)src;
    		src = (char*)src + 1;
    		dest = (char*)dest + 1;
    	}
    	return ret;
    }
    
    int main()
    {
    	int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
    	int arr2[10] = { 0 };
    	my_memcpy(arr1, arr2, 20);
    	for (int i=0; i<10; i++)
    	{
    		printf("%d ", arr1[i]);
    	}
    
    	return 0;
    }
    
  • 思路:我们注意,我们memcpy还要传递一个size_t的数,它表示的是字节
  • 如代码所示,我们是将arr2中的0给arr1中,而且是20个字节,所以是把arr1中的前五个数改为0,这是因为一个int是4个字节  arr1中的1 就表示为01 00 00 00 ,2就是 02 00 00 00 ...  这样我们就是一个一个字节的拷贝到目标空间去

 

  • memmove接收参数的含义其实和memcpy差不多,这里就不再重复了

void* my_memmove(void *dest,const void *src,size_t num)
{
	assert(dest && src);

	void* ret = dest;
	if (dest < src)
	{
		while (num--)
		{
			*(char*)dest = *(char*)src;
			dest = (char*)dest + 1;
			src = (char*)src + 1;
		}
	}
	else
	{
		while (num--)
		{
			*((char*)dest + num) = *((char*)src + num);
		}
	}
	return ret;

}

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

	return 0;
}

思路:

  • 我们要分时从前->后拷贝还是从后->前拷贝,这是因为如果dest>src的时候,如果我们传递的是my_memmpve(arr1+2,arr1,20)   这是arr1的数据是1 2 1 2 1 2 1  8 9 10,原因就是当会形成覆盖  
  • 所以,我们要分情况,因为数组是从低地址向高地址存贮的,所以当dest< src的时候,就是从前向后存储,这时我们要注意我们是一个一个字节存储的,而且我们是void* 所以我们要强转一下类型
                *(char*)dest = *(char*)src;
    			dest = (char*)dest + 1;
    			src = (char*)src + 1;

    这就是我们移动一次的时候,我们要移动num次,加上while循环就可以了。

  • 从后到前的移动是一样的思想

 


  • 最后:今天关于常见的字符串函数的模拟实现和内存的函数就到这里了,后续我也会大家多多分享一些常见的知识,多谢各位的支持了。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/742069.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于vant封装的动态表单(VFrom使用教程)

vant-ui是属于vue开发移动端中用的比较多的一个组件库了&#xff0c;网上基于它的一些组件的二次封装也数不胜数&#xff0c;但是却都是零零散散&#xff0c;不成体系。总不能用一个就去找类似的封装吧&#xff0c;这样拼凑起来的也不是我们想要的。尤其&#xff0c;涉及众多表…

Python单元测试

单元测试 测试可以保证你的代码在一系列给定条件下正常工作测试允许人们确保对代码的改动不会破坏现有的功能测试迫使人们在不寻常条件的情况下思考代码&#xff0c;这可能会揭示出逻辑错误良好的测试要求模块化&#xff0c;解耦代码&#xff0c;这是一个良好的系统设计的标志…

Oracle select 和read的权限

只有select权限&#xff0c;也可以for update锁表。 我懵了一下&#xff0c;不确认啊&#xff0c;我先测试了下&#xff0c;如下所示&#xff1a; create user test11 identified by test11; create user test12 identified by test12; grant connect,resource to test11,tes…

【2022HVV系列】蓝队手册更新版(建议收藏)

本文首发于【2022HVV系列】蓝队手册更新版&#xff08;建议收藏&#xff09; 综合篇 青藤云安全出品&#xff1a; 倒计时&#xff01;你需要这份《2022攻防演练蓝队防守指南》 攻防演练合集 | 3个阶段&#xff0c;4大要点&#xff0c;蓝队防守全流程纲要解读网络安全实战攻防…

[QT编程系列-1]:C++图形用户界面编程,QT框架快速入门培训 - 0- 总述

目录 导言 主要内容 附录&#xff1a; 导言 1. 在这里强调为啥选择 PPT 方式&#xff0c;而不是直接讲解代码 2. 重原理和方法 3. 种 QT 的框架和 QT 的开发流程 4. 轻 UI 界面美观&#xff08; UI 设计单独课程&#xff09; 5. 请代码具体实现&#xff08;后期自学&#xf…

JavaScript 导出csv文件页面卡死时的处理方式

文章概叙 当我在使用xlsx包导出csv文件时候&#xff0c;数据过大导致页面卡死&#xff0c;因此才使用blob的方式来导出csv文件&#xff0c;如果没有出现这个问题&#xff0c;建议还是使用xlsx包 XLSX包 npm的地址&#xff0c;对于前后端开发的同学来说&#xff0c;文档难度不…

【SCI一区】互联燃料电池混合动力汽车通过信号交叉口的生态驾驶双层凸优化(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

LCD显示方向

LCD显示方向 一、ILI9341内存到显示地址的映射 本文只讨论“正常显示”&#xff0c;不讨论“垂直滚动显示”模式。 可以看到物理内存被两个指针访问&#xff0c;行指针和列指针&#xff0c;行指针范围从000h到013Fh&#xff0c;列指针范围为0000h到00EFh。也就是说&#xff0c…

naive-ui 的n-date-picker改变成中文

如下 那改完之后就要变成 如下 其实蛮简单的 修改app.vue import { zhCN,dateZhCN } from "naive-ui";<n-config-provider :locale"zhCN" :date-locale"dateZhCN"><RouterView /></n-config-provider> 你要是卡到前端难题&a…

Graphics Mill 11.1.18 -24-06-2023 Crack

Graphics Mill 是适用于 .NET 和 ASP.NET 开发人员的最强大的成像工具集。它允许用户轻松向 .NET 应用程序添加复杂的光栅和矢量图像处理功能。 加载和保存 JPEG、PNG 和另外 8 种图像格式 调整大小、裁剪、自动修复、色度键和 30 多种其他图像操作 可处理任何尺寸&#xff08…

P3 第一章 电路模型与电路定律

1、在电路当中&#xff0c;理想电源的阻值与理想电源的u-i特性无关。 理想电压源的内阻是零&#xff0c;理想电流源的内阻是无穷。而在电路中的等效阻值有电路的连接情况决定。 发电机、蓄电池工作特性接近于电压源&#xff0c;电路模型表达为电压源电阻串联。 光电池特性比较…

《MySQL》内置函数

文章目录 日期函数字符串函数数学函数其他函数 日期函数 函数名称功能current_date()当前日期&#xff08;yyyy-mm-dd&#xff09;current_time()当前时间&#xff08;hh-mm-ss&#xff09;current_timestamp()当前时间戳&#xff08;yyyy-mm-dd hh-mm-ss&#xff09;now()当前…

缓存 - Spring Boot 整合 Caffeine 不完全指北

文章目录 Pre名词解释注解属性指导步骤步骤 1&#xff1a;添加依赖步骤 2&#xff1a;配置缓存步骤 3&#xff1a;使用缓存 CodepomConfigService缓存名枚举 & 常量测试入口测试 Pre 缓存 - Caffeine 不完全指北 名词解释 Cacheable&#xff1a;表示该方法支持缓存。当调…

Few-Shot Learning in Wireless Networks: A Meta-Learning Model-Enabled Scheme

1 INTRODUCTION 由于6G的复杂管理机制&#xff0c;可以在基站收集历史数据——为克服训练数据集容量的限制提供了极大的便利 问题与挑战&#xff1a; 尽管元学习可以提高网络边缘智能(network edge intelligence)的性能&#xff0c;但其在无线网络(wireless networks)中的应用…

HCIP实验---OSPF实验2

1.如图连接&#xff0c;合理规划IP地址&#xff0c;所有路由器各自创建一个loopback接口 2.R1再创建三个接口IP地址为201.1.1.1/24、201.1.2.1/24、201.1.3.1/24 R5再创建三个接口IP地址为202.1.1.1/24、202.1.2.1/24、202.1.3.1/24 R7再创建三个接口IP地址为203.1.1.1/2…

高效办公学习的秘密!打工人高效工作必备的AI神器

​人工智能 (AI) 近年来取得了显著进步&#xff0c;并已成为科技行业的流行语。我们随时能看到大量有关人工智能工具的资讯&#xff0c;它有可能自动执行任务&#xff0c;节省时间并提高效率&#xff0c;使其成为企业的宝贵资产和平台。 随着人工智能的进步&#xff0c;旨在让…

如何在Ubuntu18.04中下载MySQL5.7

检查环境 首先检查自己电脑有没有安装过mysql,输入如下&#xff1a; rpm -qa | grep mysql如果有则清理干净在安装&#xff0c;输入 whereis mysql找到文件夹目录&#xff0c;再把它删除 rpm -e --nodeps mysql-xxx检查系统是否自带mariadb&#xff0c;输入如下检查 rpm -…

基于Tars高并发IM系统的设计与实现-实战篇1

基于Tars高并发IM系统的设计与实现-实战篇1 通过前文基础篇&#xff0c;进阶篇的介绍&#xff0c;相信大家已经对IM的基础能力实现&#xff0c;难点及其如何解决有了一个全面的了解与认识。 相信大家都迫不及待想看看一个高性能的IM系统到底怎么设计与实现&#xff0c;从本篇开…

grpc学习笔记

目录 gRPC原理网络传输效率问题基本概念概览ClientServer 异步相关概念异步 Client异步 Server gRPC原理 RPC 即远程过程调用协议&#xff08;Remote Procedure Call Protocol&#xff09;&#xff0c;可以让我们像调用本地对象一样发起 远程调用。 网络传输效率问题 HTTP…

Mysql---explain详解

explain命令是用来分析查询语句的执行计划的工具。它可以显示查询语句的执行计划&#xff0c;包括查询的顺序、使用的索引、扫描的行数等信息&#xff0c;帮助开发人员优化查询语句的性能。通过分析执行计划&#xff0c;可以找到查询语句的瓶颈&#xff0c;进而优化查询语句的性…