【c语言:常用字符串函数与内存函数的使用与实现】

文章目录

  • 1. strlen函数
    • 1.1使用
    • 1.2模拟实现
  • 2.strcmp函数
    • 2.1使用
    • 2.2模拟实现
  • 3.strncmp函数
    • 3.1使用
    • 3.2模拟实现
  • 4.strcpy函数
    • 4.1 使用
    • 4.2模拟实现
  • 5.strcncpy
    • 5.1使用
    • 5.2模拟实现
  • 6.strcat函数
    • 6.1使用
    • 6.2模拟实现
  • 7.strncat函数
    • 7.1使用
    • 7.2模拟实现
  • 8.strstr函数
    • 8.1使用
    • 8.2模拟实现
  • 9.strtok函数
  • 10.strerror函数
  • 11.memcpy
    • 11.1使用
    • 11.2模拟实现
  • 12.memmove函数
    • 12.1使用
    • 12.2模拟实现
  • 13.memset函数
    • 13.1使用
    • 13.2模拟实现
  • 14.memcmp函数
    • 14.1使用
    • 14.2模拟实现

在这里插入图片描述

简介:本篇文章是对C语言中常用的字符串函数和内存函数的学习以及模拟实现
文档内容来自:https://legacy.cplusplus.com/

1. strlen函数

1.1使用

在这里插入图片描述

  • 字符串以 ‘\0’ 作为结束标志,
  • strlen函数返回的是在字符串中 ‘\0’ 前⾯出现的字符个数(不包含 '\0' )。
  • 参数指向的字符串必须要以 ‘\0’ 结束。
  • 注意函数的返回值为size_t,是无符号的( 易错
  • 使用需要包含对应的头文件<string.h>

在这里插入图片描述
下面这个代码的执行结果是什么呢?

int main()
{
	const char* str1 = "abcdef";
	const char* str2 = "bbb";
	if (strlen(str2) - strlen(str1) > 0)
	{
		printf("str2>str1\n");
	}
	else
	{
		printf("srt1>str2\n");
	}
	return 0;
}

strlen(str1) = 6,strlen(str2) = 3。3-6小于0,所以输出str>str2对吗?
恭喜你,掉坑里了

两个无符号数运算的结果还是无符号数,-3看成无符号数就是3,3>0所以输出str2>str1。

1.2模拟实现

  1. 定义新变量的方法
size_t my_strlen(char* str)
{
	size_t count = 0;
	while (*str)
	{
		count++;
		str++;
	}
	return count;
}
  1. 指针-指针
size_t my_strlen(char* str)
{
	char* cur = str;
	while (*str)
	{
		str++;
	}
	return str-cur;
}
  1. 不定义新变量,递归法
size_t my_strlen(char* str)
{
	if (*str == '\0')
	{
		return 0;
	}
	else
	{
		return 1 + my_strlen(str + 1);
	}
}

2.strcmp函数

2.1使用

在这里插入图片描述
在这里插入图片描述

两个字符串进行比较的时候就不能再像整数那样使用 > < = 来比较了,它实际比较的是两个字符串对应字符的ASCLL码值

  • 第⼀个字符串大于第⼆个字符串,则返回大于0的数字
  • 第⼀个字符串等于第⼆个字符串,则返回0
  • 第⼀个字符串小于第⼆个字符串,则返回小于0的数字

在这里插入图片描述

2.2模拟实现

int my_strcmp(const char* str1, const char* str2)
{
	assert(str1 != NULL);
	assert(str2 != NULL);
	while (*str1 == *str2)
	{
		if (*str1 == '\0')
		{
			return 0;
		}
		str1++;
		str2++;
	}
	return str1 - str2;
	/*if (*str1 > *str2)
	{
		return 1;
	}
	else
		return -1;*/
}

3.strncmp函数

3.1使用

在这里插入图片描述

strncmp函数与strcmp函数的使用十分相似,strncmp函数只是限定了比较几个字符的大小
例如:比较前五个字符的大小
在这里插入图片描述
当要比较的长度大于两个串本身的长度时,比较到其中一个串的末尾即可
在这里插入图片描述

3.2模拟实现

int my_strncmp(const char* str1, const char* str2, int num)
{
	assert(str1 != NULL);
	assert(str2 != NULL);
	//无字符串可比,直接返回0
	if (!num)
	{
		return 0;
	}
	//1.两字符串相等
	//2.在规定的范围内
	//3.字符串未到末尾
	while ((*str1 == *str2) && --num && *str1 != '\0')
	{
		str1++;
		str2++;
	}
	return *str1 - *str2;
}

4.strcpy函数

4.1 使用

在这里插入图片描述

  • 源字符串必须以 ‘\0’ 结束
  • 会将源字符串中的 ‘\0’ 拷贝到目标空间。
  • 目标空间必须足够大,以确保能存放源字符串。
  • 目标空间必须可修改。
  • 函数返回目标空间的起始地址

在这里插入图片描述

4.2模拟实现

写法1:

char* my_strcpy(char* dest, const char* src)
{
	assert(dest != NULL);
	assert(src != NULL);
	char* ret = dest;
	while (*src)
	{
		*dest = *src;
		dest++;
		src++;
	}
	*dest = *src;
	return ret;
}

写法2:

char* my_strcpy(char* dest, const char* src)
{
	assert(dest != NULL);
	assert(src != NULL);
	char* ret = dest;
	while (*dest++ = *src++)
	{
		;
	}
	return ret;
}

5.strcncpy

5.1使用

在这里插入图片描述

  • 将源字符串的前 num 个字符复制到目标字符串。
  • 如果在复制 num 个字符之前找到源 C 字符串的末尾(由 null 字符表示),则 destination 将用零填充,直到总共写入 num 个字符。(不足num个,用0填充)

在这里插入图片描述

5.2模拟实现

char* my_strncpy(char* dest, const char* src, int num)
{
	assert(dest != NULL);
	assert(src != NULL);
	char* ret = dest;
	while ((*dest++ = *src++) && --num)
	{
		;
	}
	while (--num)
	{
		*dest = 0;
		dest++;
	}
	return ret;
}

6.strcat函数

6.1使用

在这里插入图片描述

  • 源字符串必须以 ‘\0’ 结束
  • 目标字符串中也得有 \0 ,否则没办法知道追加从哪⾥开始。
  • 目标空间必须足够大,以确保能存放源字符串。
  • 目标空间必须可修改。
  • 函数返回目标空间的起始地址
  • 字符串自己给自己追加,如何?(需要使用memmove)

在这里插入图片描述

6.2模拟实现

char* my_strcat(char* dest, const char* src)
{
	assert(dest != NULL);
	assert(src != NULL);
	char* ret = dest;
	//找dest的\0
	while (*dest)
	{
		dest++;
	}
	//开始追加
	while (*dest++ = *src++)
	{
		;
	}
	return ret;
}

7.strncat函数

7.1使用

在这里插入图片描述

  • 将源的前 num 个字符附加到目标,外加一个终止 null 字符
  • 如果 source 中 C 字符串的长度小于 num,则仅复制 ‘\0’ 之前的内容
    在这里插入图片描述

7.2模拟实现

char* my_strncat(char* dest, const char* src, int num)
{
	assert(dest != NULL);
	assert(src != NULL);
	char* ret = dest;
	//找dest的\0
	while (*dest)
	{
		dest++;
	}
	//开始追加
	while (num-- && (*dest++ = *src++))
	{
		;
	}
	*dest = 0;
	return ret;
}

8.strstr函数

8.1使用

在这里插入图片描述

  • str1中找str2字符串,返回str2第一次出现位置的地址
  • 字符串的比较匹配不包含 \0 字符,以 \0 作为结束标志

在这里插入图片描述

8.2模拟实现

char* my_strstr(const char* str1, const char* str2)
{
	assert(str1 != NULL);
	assert(str2 != NULL);
	char* ptr1 = NULL;
	char* ptr2 = NULL;
	if (!(*str2))
	{
		return (char*)str1;
	}
	while (*str1)
	{
		ptr1 = (char*)str1;
		ptr2 = (char*)str2;
		while ((*ptr1 == *ptr2) && *ptr1 && *ptr2)
		{
			ptr1++;
			ptr2++;
		}
		if (*ptr2 == 0)
		{
			return (char*)str1;
		}
		str1++;
	}
	return NULL;
}

9.strtok函数

在这里插入图片描述

  • delimiters参数指向⼀个字符串,定义了用作分隔符的字符集合
  • 第⼀个参数指定⼀个字符串,它包含了0个或者多个由delimiters字符串中⼀个或者多个分隔符分割的标记。
  • strtok函数找到str中的下⼀个标记,并将其⽤ \0 结尾,返回⼀个指向这个标记的指针。(注:strtok函数会改变被操作的字符串,所以在使⽤strtok函数切分的字符串⼀般都是临时拷⻉的内容并且可修改。
  • strtok函数的第⼀个参数不为 NULL ,函数将找到str中第⼀个标记,strtok函数将保存它在字符串中的位置。
  • strtok函数的第⼀个参数为 NULL ,函数将在同⼀个字符串中被保存的位置开始,查找下⼀个标记。
  • 如果字符串中不存在更多的标记,则返回 NULL 指针。

是不是有点难理解?
用大白话说就是:

strtok函数是以一个delimiter函数中的字符为分隔依据的函数,第一次调用该函数需要将要分隔的字符串的地址传给他,它会根据分隔依据找第一次在字符串中出现该分隔符的位置,将其用\0替代,并且返回一个指向这个分隔符的指针。下一次调用就只需要传给它空指针和分隔依据,它会从上一个分隔符位置再次开始找,直到找到分隔符。最终找到字符串结尾即可。

在这里插入图片描述

10.strerror函数

strerror函数可以把参数部分错误码对应的错误信息的字符串地址返回来。

  • 在不同的系统和C语⾔标准库的实现中都规定了⼀些错误码,⼀般是放在 errno.h 这个头⽂件中说明的,C语⾔程序启动的时候就会使⽤⼀个全⾯的变量errno来记录程序的当前错误码,只不过程序启动的时候errno是0, 表⽰没有错误,
  • 当我们在使⽤标准库中的函数的时候发⽣了某种错误,就会讲对应的错误码,存放在errno中,⽽⼀个错误码的数字是整数很难理解是什么意思,所以每⼀个错误码都是 有对应的错误信息的。

strerror函数就可以将错误对应的错误信息字符串的地址返回。

打印0-10错误码对应的信息:
在这里插入图片描述
例如:我们使用fopen函数打开本地某一个文件夹,它就会自动打印处使用该函数产生的错误。
在这里插入图片描述
再我们以上所讲的所有字符串函数,你要相对字符串进行操作,是不是必须知道它的类型呀?可不是什么时候我们都能准确的知道的,比如一个结构体类型的数据。当我们不知道的时候,那么该如何操作呢?下面我们将进行讲解:

11.memcpy

11.1使用

在这里插入图片描述

  • 将 num 字节的值从源指向的位置直接复制到目标指向的内存块
  • 源指针和目标指针指向的对象的基础类型与此函数无关;结果是数据的二进制副本
  • 该函数不检查源中的任何终止 null 字符 - 它始终准确复制 num 个字节

在这里插入图片描述

11.2模拟实现

void* my_memcpy(void* dest, const void* src, int num)
{
	void* ret = dest;
	while (num--)
	{
		*(char*)dest = *(char*)src;
		dest = (char*)dest + 1;
		src = (char*)src + 1;
	}
	return ret;
}

12.memmove函数

12.1使用

在这里插入图片描述

  • 和memcpy的差别就是memmove函数处理的源内存块和⽬标内存块是可以重叠的。
  • 如果源空间和⽬标空间出现重叠,就得使⽤memmove函数处理

在这里插入图片描述

12.2模拟实现

void* my_memmove(void* dest, const void* src, int num)
{
	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;
}

13.memset函数

13.1使用

在这里插入图片描述

  • memset是⽤来设置内存的,将内存中的值以字节为单位设置成想要的内容。
  • 将ptr指向的内存的第一个字节设置为指定的值

在这里插入图片描述

13.2模拟实现

void* my_memset(void* ptr, int x, int num)
{
	void* ret = ptr;
	while (num--)
	{
		*((char*)ptr + num) = x;
	}
	return ret;
}

14.memcmp函数

14.1使用

在这里插入图片描述
在这里插入图片描述

  • 比较从ptr1和ptr2指针指向的位置开始,向后的num个字节,
  • 如果它们都匹配,则返回零,或者返回一个不同于零的值,如果它们不匹配,则表示哪个值更大。
  • 请注意,与 strcmp 不同,该函数在找到 null 字符后不会停止比较,

在这里插入图片描述

14.2模拟实现

/*
当buf1<buf2时,返回值<0
当buf1=buf2时,返回值=0
当buf1>buf2时,返回值>0
*/
int my_memcmp(const void* buffer1, const void* buffer2, int num)
{
	 当比较位数不为0时,且每位数据相等时,移动指针
	while (--num  && *(char*)buffer1 == *(char*)buffer2)
	{
		buffer1 = (char*)buffer1 + 1;
		buffer2 = (char*)buffer2 + 1;
	}
	return (*(char*)buffer1) - (*(char*)buffer2);
}

目前本人学习和使用到的字符串、内存函数就这些,如有错误,请批评指正!
在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

ffmpeg 免安装,配置环境变量

1、下载ffmpeg https://download.csdn.net/download/qq284489030/88579595 2、解压 解压ffmpeg-4.4-essentials_build.zip到目标文件夹&#xff0c;比如 d:\apps下&#xff1b; 3、配置环境变量 &#xff08;1&#xff09;电脑桌面鼠标右键点击“此电脑”&#xff0c;弹出…

[带余除法寻找公共节点]二叉树

二叉树 题目描述 如上图所示&#xff0c;由正整数1, 2, 3, ...组成了一棵无限大的二叉树。从某一个结点到根结点&#xff08;编号是1的结点&#xff09;都有一条唯一的路径&#xff0c;比如从10到根结点的路径是(10, 5, 2, 1)&#xff0c;从4到根结点的路径是(4, 2, 1)&#x…

C++算法入门练习——数据流第K大元素

现有一个初始为空的序列S&#xff0c;对其执行n个操作&#xff0c;每个操作是以下两种操作之一&#xff1a; 往序列S中加入一个正整数x&#xff1b;输出当前序列S​中第k​大的数。 其中&#xff0c;第k大是指将序列从大到小排序后的第k个数。 利用stl里的priority_queue自动…

如何让电脑每天定时自动关机?

如何让电脑每天定时自动关机&#xff1f;电脑已经成为社会生产活动中不可或缺的一种工具&#xff0c;它对于我们每个人都非常的重要&#xff0c;不管是工作、生活还是学习中&#xff0c;我们都需要利用电脑。不过很多小伙伴因为繁忙或者因为其它的事情&#xff0c;导致电脑经常…

Vue3水印(Watermark)

APIs 参数说明类型默认值必传width水印的宽度&#xff0c;默认值为 content 自身的宽度numberundefinedfalseheight水印的高度&#xff0c;默认值为 content 自身的高度numberundefinedfalserotate水印绘制时&#xff0c;旋转的角度&#xff0c;单位 number-22falsezIndex追加…

oracle官方的反解析工具:javap详解

1、解析字节码的作用 通过反编译生成的字节码文件&#xff0c;我们可以深入的了解java代码的工作机制。但是&#xff0c;自己分析类文件结构太麻烦了&#xff01;除了使用第三方的jclasslib工具之外&#xff0c;oracle官方也提供了工具&#xff1a;javap javap是jdk自带的反解…

数据结构与算法之美学习笔记:28 | 堆和堆排序:为什么说堆排序没有快速排序快?

目录 前言如何理解“堆”&#xff1f;如何实现一个堆&#xff1f;1. 往堆中插入一个元素2. 删除堆顶元素 如何基于堆实现排序&#xff1f;1. 建堆2. 排序 解答开篇内容小结 前言 本节课程思维导图&#xff1a; 我们今天讲另外一种特殊的树&#xff0c;“堆”&#xff08;Heap&…

【蓝桥杯选拔赛真题69】Scratch洗牌发牌 少儿编程scratch图形化编程 蓝桥杯创意编程选拔赛真题解析

目录 scratch洗牌发牌 一、题目要求 编程实现 二、案例分析 1、角色分析

2023年大数据场景智能运维实践总结

作者&#xff1a;放纵 引言 在当今数字化世界中&#xff0c;如何充分挖掘和发挥数据价值已经成为了企业成功的关键因素&#xff0c;大数据也成为企业决策和运营的重要驱动力。在《当我们在谈论DataOps时&#xff0c;我们到底在谈论什么》一文中也提到&#xff0c;企业在面对到…

P8A004-系统加固-磁盘访问权限

【预备知识】 访问权限&#xff0c;根据在各种预定义的组中用户的身份标识及其成员身份来限制访问某些信息项或某些控制的机制。访问控制通常由系统管理员用来控制用户访问网络资源&#xff08;如服务器、目录和文件&#xff09;的访问&#xff0c;并且通常通过向用户和组授予…

数据结构与算法--特殊的完全二叉树--堆,堆排序,利用堆解决topk的问题

目录 前言 1.树概念及结构 1.1树的概念 1.2 树的相关概念 1.3 树的表示 1.4 树在实际中的运用&#xff08;表示文件系统的目录树结构&#xff09; 2.二叉树概念及结构 2.1概念 2.2现实中的二叉树&#xff1a; 2.3 特殊的二叉树&#xff1a; 2.4 二叉树的性质 …

网络协议系列:TCP三次握手,四次挥手的全过程,为什么需要三次握手,四次挥手

TCP三次握手&#xff0c;四次挥手的全过程&#xff0c;为什么需要三次握手&#xff0c;四次挥手 一. TCP三次握手&#xff0c;四次挥手的全过程&#xff0c;为什么需要三次握手&#xff0c;四次挥手前言TCP协议的介绍三次握手三次握手流程&#xff1a;1. A 的 TCP 向 B 发送 连…

Windows关闭端口服务命令

winR 打开命令运行 cmd 命令netstat -o -n -a | findstr :9993 显示所有的端口占用情况 -a 显示所有连接和监听端口 -n 以数字形式显示地址和端口号。 此选项一般与 -a选项组合使用 -o 显示与每个连接相关的所属进程 ID 终止 PID taskkill /F /PID 3652

从独立求存到登顶市场,荣耀为何能在手机红海翻出新的浪花?

对企业的价值评估&#xff0c;往往离不开对其所处行业前景的考量。在蓝海赛道布局的企业&#xff0c;往往要比在红海市场突围的企业更容易受到资本重视。 但这并非绝对&#xff0c;若是一家企业能够在饱和的红海市场中&#xff0c;实现新的增长&#xff0c;其蕴涵的成长价值便…

Influx集群解决方案(Influx Proxy篇)

InFluxDB 集群搭建 本次搭建使用influx proxy 介绍 github地址:https://github.com/chengshiwen/influx-proxy/ Influx Proxy 是一个基于高可用、一致性哈希的 InfluxDB 集群代理服务&#xff0c;实现了 InfluxDB 高可用集群的部署方案&#xff0c; 具有动态扩/缩容、故障恢复…

字符串函数精讲1

又是好几天没有更新了&#xff0c;最近有些忙&#xff0c;但这并不是理由&#xff0c;还是怪我自己玩的时间多了&#xff01;但还是有在每天敲代码的&#xff01;话不多说&#xff0c;开始这一期的学习&#xff1a; strlen的使用和模拟实现 • 字符串以 \0 作为结束标志&#…

java学习part24异常throws

127-异常处理-异常处理方式二&#xff1a;throws_哔哩哔哩_bilibili 1.方法throws 2.如何抉择try和throws 3.手动throw语句 抛出一些java语法上没错但是不符合实际情况的异常。 用throw手动抛&#xff0c;方法上必须加throws。除非是运行时异常。 4.自定义异常

Java常见CodeReview及编码规范

鉴于自己的开发经验,以及常见容易产生bug及性能问题的点做个记录. 1.数据库 如果开发人员的经验不足,Java通过ORM(Mybatis)对数据库的操作的性能问题比较隐蔽.因为不压测或者异常case没发生的时候一般发现不了问题.特别是异常case发生的时候. 除配置表以外的sql都要经过expl…

软件设计师——程序设计语言基础(一)

&#x1f4d1;前言 本文主要是【程序设计语言基础】——程序设计语言基础的相关题目&#xff0c;如果有什么需要改进的地方还请大佬指出⛺️ &#x1f3ac;作者简介&#xff1a;大家好&#xff0c;我是听风与他&#x1f947; ☁️博客首页&#xff1a;CSDN主页听风与他 &#…

荣耀冲击高端,一边推新「修路」,一边降价「拆桥」

作者 | 辰纹 来源 | 洞见新研社 从2020年11月17日与华为分家&#xff0c;开启独立创业之路&#xff0c;到成功逆袭&#xff0c;今年第三季度以18%的份额重回中国智能手机市场榜首&#xff0c;荣耀用了3年时间。 图源&#xff1a;Canalys 在这三年时间内&#xff0c;荣耀经历…