【C语言】字符串函数strcpystrcatstrcmpstrstr的使⽤和模拟实现

news2024/11/29 6:25:35
  🌈write in front :

🔍个人主页 : @啊森要自信的主页

✏️真正相信奇迹的家伙,本身和奇迹一样了不起啊!

欢迎大家关注🔍点赞👍收藏⭐️留言📝>希望看完我的文章对你有小小的帮助,如有错误,可以指出,让我们一起探讨学习交流,一起加油鸭。 请添加图片描述

文章目录

  • 📝前言
  • 🌠 库函数strcpy
    • 🌉strcpy的模拟实现
  • 🌠库函数strcat
    • 🌉strcat的模拟实现
  • 🌠strcmp 的使⽤
    • 🌉strcmp 模拟实现
  • 🌠 strstr 的使⽤
    • 🌉strstr 的模拟实现
  • 🚩总结


📝前言

记上节,我们学了字符串strlen的使用和三种模拟实现方法,本小节,阿森继续和你一起学习4个字符串函数:strcpystrcatstrcmpstrstr的使用和他的模拟实现方法,学习这些库函数,可以更好的方便操作字符和字符串,文章干货满满,接下来我们就学习一下这些函数吧!


🌠 库函数strcpy

strcpy函数是将源字符串拷贝到目标字符串中,覆盖目标字符串原有内容。

char *strcpy(char *dest, const char *src);
dest:目标字符串,用于保存拷贝结果。
src:源字符串,将其内容拷贝到dest中。
返回值:返回目标字符串dest的指针。

注意点:

  • strcpy函数会将src字符串中的字符一个一个地复制到dest字符串中,直到遇到字符串结束标志'\0'为止。

  • src字符串必须以'\0'结束。

  • 会将源字符串中的 '\0' 拷⻉到⽬标空间。

  • dest字符串要有足够的空间,不然会导致缓冲区溢出。
    在这里插入图片描述

  • 目标空间必须是可修改的
    在这里插入图片描述

代码:

int main()
{
	char arr1[20] = "xxxxxxxxx";
	char arr2[] = {'a', 'b', 'c','\0'};
	strcpy(arr1, arr2);
	printf("%s\n", arr1);

	return 0;
}

运行结果:
在这里插入图片描述

🌉strcpy的模拟实现

  1. 方法一
void my_strcpy(char* dest, char* src)
{
	while (*src != '\0')
	{
		*dest = *src;
		dest++;
		src++;
	}
	*dest = *src;
}
int main()
{
	char arr1[20] = { 0 };
	char arr2[] = "abcdef";
	my_strcpy(arr1, arr2);
	printf("arr1 after copy: %s\n", arr1);
	return 0;
}

分析: 使用while循环,循环条件是源字符串src指针指向的字符不为'\0',每次循环体内,将src指向字符复制到dest指向位置,分别使destsrc指针后移,指向下一个字符位置,循环结束后,将字符串结束符'\0'也复制到dest指向位置

  1. 方法二
    第一种方法缺陷太多了,比如没有返回值,无法知道拷贝是否成功。
char* my_strcpy(char* dest, const char* src)
{
	assert(dest != NULL);
	assert(src != NULL);
	检查dest和src参数是否为NULL,NULL参数会导致访问异常。需要#include <assert.h>
	
	char* ret = dest;
	保存dest原始地址到ret变量,后面返回值使用。
	
	assert(dest && src);
	再次检查dest和src是否合法,防御性编程。
	
	while (*dest++ = *src++)
	{
		;
	}
	return ret;
}

while (*dest++ = *src++)
先一次赋值destsrc当前字符(*dest = *src),然后并使指针后(dest=dest+1,src=src+1),while循环结束条件是当src字符串末尾'\0'字符被复制时结束,当src指向\0,此时\0赋值给*dest,符合while循环结束条件,跳出循环.

🌠库函数strcat

strcat函数用于将源字符串src追加到目标字符串dest的结尾。

char *strcat(char *dest, const char *src);
dest - 目标字符串,必须有足够的空间存储追加后的结果字符串。
src - 源字符串。
返回值: 返回目标字符串dest的地址。

strcat首先找到dest字符串的结尾,即'\0'字符的位置。从dest字符串结尾开始,将src字符串一个字符一个字符地复制过来。将src字符串的结束'\0'字符也复制到dest后面。最后返回dest地址。

关键点:

  • 源字符串src必须以 '\0' 结束。
  • ⽬标字符串中也得有 \0 ,否则没办法知道追加从哪⾥开始。
  • dest必须有足够的空间存储追加后的结果字符串。
  • ⽬标空间必须可修改。
  • 字符串⾃⼰可不可以给⾃⼰追加?
    使用:
int main()
{
	char str1[20] = "Hello ";
	char str2[20] = "World";
	char str3[40];

	/* 复制str1到str3 */
	strcpy(str3, str1);

	/* 将str2追加到str3结尾 */
	strcat(str3, str2);

	printf("str3 = %s\n", str3);

	return 0;
}

输出:

str3 = Hello World
在这里插入图片描述

🌉strcat的模拟实现

  1. strcat的模拟实现一
char* my_strcat(char* dest, const char* src)
{
	char* ret = dest;
	assert(dest && src);检查参数是否合法。
	//1. 找到目标空间的\0
	使用while循环找到dest字符串的结束'\0'字符,dest指针指向字符串结束后的位置。
	while (*dest)
	{
		dest++;
	}
	//2. 拷贝
	while (*dest++ = *src++)
	{
		;
	}

	return ret;
}

while (*dest++ = *src++)先一次赋值destsrc当前字符(*dest = *src),然后并使指针后(dest=dest+1,src=src+1),先读取src的一个字符将字符赋给dest指向的位置,然后destsrc同时后移一个位置循环,当src指向\0,此时\0赋值给*dest,循环结束.

2.字符串⾃⼰可不可以给⾃⼰追加? 答案是:不能给自己追加

 char* my_strcat(char* dest, const char* src)
{
	char* ret = dest;
	assert(dest && src);
	//1. 找到目标空间的\0
	while (*dest)
	{
		dest++;
	}
	//2. 拷贝
	while (*dest++ = *src++)
	{
		;
	}

	return ret;
}

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

	return 0;
}

运行代码图:
在这里插入图片描述
在这里插入图片描述

🌠strcmp 的使⽤

strcmp用于比较两个字符串是否相等,也就是比较字符串大小的函数。
函数原型:

int strcmp(const char *str1, const char *str2);
str1和str2是要比较的两个字符串指针。

strcmp比较字符串的大小,不是按字符串的长度进行比较,而是逐个字符地比较两个字符串对应的每个字符的ASCII码值。(比较使用的是无符号字符值的ASCII码顺序。)

  • 返回值:

    • 如果str1str2完全相等,返回0

    • 如果str1大于str2(按ASCII码顺序),返回一个大于0的数。

    • 如果str1小于str2,返回一个小于0的数。

  • strcmp()函数是C标准库string.h头文件中的函数。

  • 字符串比较结束条件是遇到字符串末尾'\0'字符或者第一个不匹配字符。

字符串"cat" 和 "dog" 的比较:
'c'的ASCII码是99, 'd'的ASCII码是100,所以"cat"小于"dog"

字符串"hello" 和 "hello world" 的比较: 
前6个字符都相等,但第7个字符' '的ASCII码小于'\0',所以"hello"小于"hello world"

例子:

#include <string.h>

int main()
{
  char str1[] = "ahbyb";
  char str2[] = "asyzx";  

  int result = strcmp(str1, str2);

  if(result == 0)
    printf("Strings are equal\n");
  else if(result > 0)  
    printf("str1 is greater than str2\n"); 
  else
    printf("str1 is less than str2\n");

  return 0;
}

结果:

str1 is less than str2

在这里插入图片描述

🌉strcmp 模拟实现

int my_strcmp(const char* s1, const char* s2)
{
	while (*s1 == *s2)
	{
		if (*s1 == '\0')
			return 0;
		s1++;
		s2++;
	}
	return *s1 - *s2;
		//if (*s1 > *s2)
	//	return 1;
	//else
	//	return -1;

}

使用while循环逐个比较s1s2每个字符是否相等如果字符相等,继续循环比较下一个字符, 如果遇到字符串结束符’\0’,表示两个字符串完全匹配,直接返回0,如果在循环中找到不匹配的字符,使用*s1 - *s2返回两个字符的ASCII码差值

🌠 strstr 的使⽤

strstr用来查找一个字符串在另一个字符串中首次出现的位置。

strstr函数的原型:

char* strstr(const char* str1, const char* str2);
- str1: 主字符串,要在其中查找子字符串
- str2: 子字符串,要查找的字符串

strstr函数可以用来在一个字符串中查找另一个字符串首次出现的位置,如果str2不存在于str1中,则返回NULL;如果str2存在于str1中,则返回第一个匹配位置的指针。

strstr的比较原理是:

  1. str1字符串的起始位置开始,与str2字符串进行字符匹配比较。

  2. 如果匹配失败(当前字符不同),则str1指针后移一位,继续匹配。

  3. 如果匹配成功(到达str2字符串结束符'\0'),则匹配成功,返回str1指针地址。

  4. 如果遍历完str1仍未匹配成功,则返回NULL

例如:

char* p = strstr("hello world","world");
// p指向"world"子字符串在"hello world"中的位置

这里用一个图来解释strstr函数的工作原理:

        +----------------------+
str1 => | h e l l o   w o r l d| 
        +----------------------+
              |
              V
        +-----------+
str2 => | w o r l d |
        +-----------+
              |
              V
         比较第一个字符'h'与'w',不匹配
              |
              V
       指针后移到下一个字符'e'
              |
              V
         比较'e'与'w',不匹配
              |
              V
       指针后移到下一个字符'l'
              |  
              V
         比较'l'与'w',不匹配
              |
              V
       指针后移,依次比较直到匹配成功
              |
              V
         当str1指针指向'w'时,与str2第一个字符'w'匹配
              |
              V
       开始匹配后续字符,全部匹配成功
              |
              V
       返回str1指针地址,指向子字符串在主字符串中的位置
int main()
{
	char arr1[] = "abcegtbaab";
	char arr2[] = "cegtba";
	char* ret = strstr(arr1, arr2);
	if (ret != NULL)
		printf("%s\n", ret);
	else
		printf("找不到\n");

	return 0;
}

结果:

cegtbaab

当然也可以用图展示:
在这里插入图片描述

strstrstr1起始位置开始,用str2str1进行字符匹配比较。如果不匹配就后移str1指针,匹配成功就返回str1当前位置指针,上图就是返回c的地址。通过这种逐个匹配的方式找到子字符串在主字符串中的第一个匹配位置。

🌉strstr 的模拟实现

char* my_strstr(const char* str1, const char* str2)
{
	const char* cur = str1;//用cur记录str1的位置
	const char* s1 = NULL;//使用assert检查str1和str2是否为非空指针。
	const char* s2 = NULL;

	assert(str1 && str2);
	if (*str2 == '\0')//检查str2是否为空字符串,如果为空直接返回str1。
	{
		return (char*)str1;
	}

	while (*cur)//使用cur指针遍历str1。
	{ //每次遍历:
		s1 = cur;//将cur赋值给s1,将str2赋值给s2,用于后续匹配
		s2 = str2;//当然,第几次失败后,重新回溯,重新开始匹配
		while (*s1 && *s2 && *s1 == *s2)
		{
			s1++;//开始匹配s1和s2中的字符,同时递增s1和s2。
			s2++;
		}
		if (*s2 == '\0')如果s1和s2匹配到结尾('\0'),表示找到了子串,返回cur。
		{
			return (char*)cur;
		}
		cur++;匹配失败后,cur++继续下次匹配。
	}
	return NULL;遍历完str1没有找到匹配,返回NULL。
}

时间复杂度为O(MN),其中M和N分别为主串和子串的长度。
若老铁们有点蒙蒙的,可以结合下图来理解:
在这里插入图片描述


🚩总结

这次阿森和你一起学习4个C语言中常用的基本字符操作函数,当然这只是一部分,还有很多,但阿森会慢慢和你一起学习。感谢你的收看,如果文章有错误,可以指出,我不胜感激,让我们一起学习交流,如果文章可以给你一个小小帮助,可以给博主点一个小小的赞😘
请添加图片描述

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

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

相关文章

聚类算法的性能度量

聚类算法的性能度量 聚类算法就是根据数据中样本与样本之间的距离或相似度&#xff0c;将样本划分为若干组&#xff0f;类&#xff0f;簇&#xff0c;其划分的原则&#xff1a;簇内样本相似、簇间样本不相似&#xff0c;聚类的结果是产生一个簇的集合。 其划分方式主要分为两…

前后端联调神器《OpenAPI-Codegen》

在后端开发完接口之后&#xff0c;前端如果再去写一遍接口来联调的话&#xff0c;会很浪费时间&#xff0c;这个时候使用OpenAPI接口文档来生成Axios接口代码的话&#xff0c;会大大提高我们的开发效率。 Axios引入 Axios是一个基于Promise的HTTP客户端&#xff0c;用于浏览器…

leetcode每日一题39

122.买卖股票的最佳时机II 确定dp数组&#xff08;dp table&#xff09;以及下标的含义 dp[i][j] j0 不持有股票 j1持有股票 i&#xff1a;第i天 dp[i][j]&#xff1a;第i天&#xff0c;持有状态为j时的最大现金确定递推公式 dp[i][0]max(dp[i-1][0], dp[i-1][1]prices[i]); 第…

Linux权限(用户角色+文件权限属性)

Linux权限 文章目录 Linux权限一.文件权限1.快速掌握修改权限的方法&#xff08;修改文件权限属性&#xff09;2.对比权限的有无&#xff0c;以及具体的体现3.修改权限的第二套方法&#xff08;修改用户角色&#xff09;4.文件类型&#xff08;Linux下一切皆文件&#xff09; 二…

编辑拒稿理由是重复率高

大家好&#xff0c;今天来聊聊编辑拒稿理由是重复率高&#xff0c;希望能给大家提供一点参考。 以下是针对论文重复率高的情况&#xff0c;提供一些修改建议和技巧&#xff1a; 编辑拒稿理由是重复率高 当作者提交论文到学术期刊后&#xff0c;编辑会对论文进行审核小发猫写作…

Vue3: 给表格多个字段添加排序功能

问题 在Vue3项目中&#xff0c;使用element-plus的表格组件绘制表格后&#xff0c;需要令表格的多个字段可以进行选择排序&#xff08;选择升序或者降序&#xff09;但是排序功能好像有时候会出错&#xff0c;需要排序的字段多了之后&#xff0c;排序功能有时候会不起作用 解…

算法专题一:双指针

算法专题一&#xff1a;双指针 一&#xff1a;移动零1.GIF题目解析&#xff1a; 二&#xff1a;复写零2.GIF题目解析&#xff1a; 三&#xff1a;快乐数3.GIF题目解析&#xff1a; 四&#xff1a;装水最多容器&#xff1a;4.GIF题目解析&#xff1a; 五&#xff1a;有效三角形的…

知识产权服务企业网站建设效果如何

知识产权服务也有较高的市场需求度&#xff0c;尤其如今互联网深入到各个行业&#xff0c;无论个人还是企业都会以不同的方式经营&#xff0c;相应的为保障自身权益&#xff0c;注册商标、专利等自然不可少&#xff0c;而对普通小白来说&#xff0c;想要完成这些流程也是有些难…

NSSCTF-Crypto靶场练习--第11-20题wp

文章目录 [SWPUCTF 2021 新生赛]traditional[LitCTF 2023]梦想是红色的 (初级)[SWPUCTF 2021 新生赛]crypto2[羊城杯 2021]Bigrsa[LitCTF 2023]Hex&#xff1f;Hex&#xff01;(初级)[SWPU 2020]happy[AFCTF 2018]BASE[安洵杯 2019]JustBase[鹤城杯 2021]Crazy_Rsa_Tech[SWPUCT…

mybatisPlus框架

1、特性 无侵入 &#xff1a;只做增强不做改变&#xff0c;引入它不会对现有工程产生影响&#xff0c;如丝般顺滑 损耗小 &#xff1a;启动即会自动注入基本 CURD &#xff0c;性能基本无损耗&#xff0c;直接面向对象操作 强大的 CRUD 操作 &#xff1a;内置通用 Mapper 、…

Oracle(2-12)User-Managed Complete Recovery

文章目录 一、基础知识1、Media Recovery 介质恢复2、Recovery Steps 恢复步骤3、恢复4、Recovery in ARCHIVELOG 在ARCHIVELOG中恢复5、Complete Recovery完全恢复6、CR in ARCHIVELOG Mode 归档日志模式下的完全恢复7、Determine Files Need Recovery确定需要恢复的文件8、Ab…

JDK8新特性:Lambda表达式规则及用法,方法引用

目录 Lambda表达式是JDK8新增的一种语法格式 1.作用 2.用法规则&#xff1a; 3.方法引用 Lambda表达式是JDK8新增的一种语法格式 1.作用 简化匿名内部类的代码写法 Lambad用法前提&#xff1a;只能简化函数式接口&#xff08;一般加有Funcationallnterface&#xff09;&a…

2023年11月10日 Go生态洞察:十四年Go的成长之路

&#x1f337;&#x1f341; 博主猫头虎&#xff08;&#x1f405;&#x1f43e;&#xff09;带您 Go to New World✨&#x1f341; &#x1f984; 博客首页——&#x1f405;&#x1f43e;猫头虎的博客&#x1f390; &#x1f433; 《面试题大全专栏》 &#x1f995; 文章图文…

STL(八)(总结篇)

###以四道题来总结 题号:lanqiao OJ 3226 1.宝藏排序II ### 这道题主要考察sort,非常简单输出就是升序不需要自定义比较函数 #include<bits/stdc.h> using namespace std; const int N1e55; //这里用int就足够了不需要开long long int a[N]; int main(){ios::sync_with…

TCP的滑动窗口机制

网络的错误检测和补偿机制非常复杂。 一、等待超时时间&#xff08;返回ACK号的等待时间&#xff09; 当网络繁忙时会发生拥塞&#xff0c;ACK号的返回变慢&#xff0c;较短的等待时间会导致频繁的数据重传&#xff0c;导致本就拥塞的网络雪上加霜。如果等待时间过长&#xf…

查看mysql是否开启远程端口

这个命令&#xff1a; sudo netstat -tlnp | grep mysqld如果是 就说明只开启了本地的&#xff0c;要更改这个设置&#xff0c;你需要编辑 MySQL 的配置文件&#xff0c;并确保 bind-address 设置为 0.0.0.0。打开 MySQL 的配置文件&#xff08;通常是 /etc/mysql/mysql.conf…

二叉排序树的判断(二叉树的顺序存储):2022年408算法题

对于采用顺序存储方式保存的二叉树&#xff0c;根结点保存在SqBiTNode[0]中&#xff1b;当某结点保存SqBiTNode[i]中时&#xff0c;若有左孩子&#xff0c;则其值保存在SqBiTNode [2i1]中&#xff1b;若有右孩子&#xff0c;则其值保存在SqBiTNode[2i2]中&#xff1b;若有双亲结…

SD之lora训练

目录 为什么要训练自己的模型 SD模型微调方法 准备素材 1 确定要训练的LoRA类型 2 图片收集 3 图片预处理 4 图片标注 安装Koyha_ss 训练lora 1.准备参数和环境 2.启动训练 使用模型 1 拷贝训练过的lora模型 2 启动SD WebUI进行图像生成 为什么要训练自己的模型 …

来聊聊java8的数值流

简介 java8为我提供的简单快捷的数值流计算API&#xff0c;本文就基于几个常见的场景介绍一下数值流API的使用。 基础示例 我们以一个食物热量计算的功能展开演示&#xff0c;如下所示&#xff0c;可以看到Dish类它记录了每一个食物的名称、热量、类型等信息: public class…

百科词条可以删除吗?如何删除自己的百度百科?

近日&#xff0c;小马识途营销顾问接到不少客户删除自己百科词条的咨询&#xff0c;有不少人自己并没有去建立百科词条&#xff0c;但是网上已经有了&#xff0c;有的信息不正确&#xff0c;甚至有的信息是负能量的&#xff0c;对当事人自己造成一定的困扰&#xff0c;所以寻求…