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

  🌈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. 找到目标空间的\0while (*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"); elseprintf("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);elseprintf("找不到\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.hqwc.cn/news/263518.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

一文1800字从0到1使用Python Flask实战构建Web应用

Python Flask是一个轻量级的Web框架&#xff0c;它简单易用、灵活性高&#xff0c;适用于构建各种规模的Web应用。本文将介绍如何使用Python Flask框架来实战构建一个简单的Web应用&#xff0c;并展示其基本功能和特性。 第一部分&#xff1a;搭建开发环境 在开始之前我们需要…

基于Java SSM框架实现高校毕业生就业满意度调查平台项目【项目源码+论文说明】

基于java的SSM框架实现高校毕业生就业满意度调查平台演示 摘要 随着科学技术的飞速发展&#xff0c;社会的方方面面、各行各业都在努力与现代的先进技术接轨&#xff0c;通过科技手段来提高自身的优势&#xff0c;高校毕业生就业满意度调查统计系统当然也不能排除在外。高校毕…

王道数据结构课后代码题 p149 第8—— 12(c语言代码实现)

目录 8.假设二叉树采用二叉链表存储结构存储&#xff0c;试设计一个算法&#xff0c;计算一棵给定二叉树的所有双分支结点个数。 9.设树B是一棵采用链式结构存储的二叉树&#xff0c;编写一个把树 B中所有结点的左、右子树进行交换的函数。 10.假设二叉树采用二叉链存储结构存储…

[全志Tina/Linux]全志在线生成bootlogo工具

一、需求 由于全志的bootlogo文件要求使用bmp格式的32位RGBA图像&#xff0c;经测试在使用不同版本的ps软件修图时&#xff0c;导出的bootlogo.bmp经常无法被全志uboot识别&#xff0c;因此使用在线工具转换。 二、操作 1、登录工具网站 https://online-converting.com/ima…

git 本地有改动,远程也有改动,且文件是自动生成的配置文件

在改动过的地方 文件是.lock文件&#xff0c;自动生成的。想切到远程的分支&#xff0c;但是远程的分支也有改动过。这时候就要解决冲突&#xff0c;因为这是两个分支&#xff0c;代码都是不一样的&#xff0c;要先把这改动的代码提交在本地或者提交在本分支的远程才可以切到其…

51系列--基于MPX4250的压力计仿真设计

本文介绍基于MPX4250的压力计仿真设计&#xff08;完整仿真源文件及代码见文末链接&#xff09; MPX4250是一种线性度极强的一种压力传感器&#xff0c;它的线性范围为20-250kpa,测量范围也比较广泛&#xff0c;适用于大多数场合的压力检测。 仿真图如下 仿真运行视频 51系列…

Rsync+Sersync

服务器相关参数 源服务器 192.168.17.101 目标服务器&#xff08;同步到的服务器&#xff09; 192.168.17.103 ##目标服务器配置 ###1、配置rsync服务 1、安装rsync yum -y install rsync 2、配置rsync vim /etc/rsyncd.conf 配置文件内容 uid root gid root use c…

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

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

Leetcode每日一题

https://leetcode.cn/problems/binary-tree-preorder-traversal/ 这道题目需要我们自行进行创建一个数组&#xff0c;题目也给出我们需要自己malloc一个数组来存放&#xff0c;这样能达到我们遍历的效果&#xff0c;我们来看看他的接口函数给的是什么。 可以看到的是这个接口函…

笔记本电脑word打字延迟特别大,但是浏览器中打字没有延迟,如何解决这个问题。

问题描述&#xff1a; 笔记本电脑word打字延迟特别大&#xff0c;但是浏览器中打字没有延迟&#xff0c;如何解决这个问题。&#xff08;之前以为是自己的电脑用了6年&#xff0c;用的时间久了&#xff0c;硬件老化导致的&#xff0c;本来想直接换电脑的&#xff0c;但是想着去…

C语言指针基础题(一)

目录 例题一题目解析答案 例题二题目解析答案 例题三题目解析答案 例题四题目解析答案 例题五题目解析答案 例题六题目解析答案 例题七题目解析答案 感谢各位大佬对我的支持,如果我的文章对你有用,欢迎点击以下链接 &#x1f412;&#x1f412;&#x1f412; 个人主页 &#x…

简易加减运算器的制作----数字电路设计(含proteus仿真)

简易加减运算器的制作 一、功能要求—基本功能 1、自制0-9按键&#xff0c;在一个LED数码管上稳定地显示当前按下的值。&#xff08;基本功能&#xff09; 2、增加、两个按键&#xff0c;实现0-9两个一位数的加法运算&#xff0c;同时在两位LED上稳定地显示运算结果。&#…