【C语言】——字符串函数的使用与模拟实现(上)

news2024/11/20 8:23:00

【C语言】——字符串函数

前言

  在我们学习 C语言 的过程中,对库函数的使用是必不可少的。其中,相信大家最熟悉的就是 < s t d i o . h > <stdio.h> <stdio.h> 中的 p r i n t print print s c a n f scanf scanf 函数了吧。
  但是今天,我们不讲他们(太难了,呜呜呜),今天我们来讲与字符串相关的函数,也就是 < s t r i n g . h string.h string.h> 中的系列函数。
  要知道,除了输入输出,我们对字符串的操作也是必不可少的,那么了解字符串相关的函数的重要性就不言而喻了,让我们开启接下来的学习吧。

一、 s t r l e n strlen strlen 函数

1.1、函数功能

  • 函数声明:size_t strlen(const char* str);
  • s t r l e n strlen strlen 函数用于求字符串的长度,使用时必须包含头文件
  • 基本原理是:从传递的指针变量开始,从前往后计算字符的个数,直到遇到 ‘\0’ 停止‘\0’ 本身不计算),因此参数指向的字符串必须要以 ‘\0’ 结束
  • 返回值为字符串的长度,注意:返回值是 size_t 类型(易错)

  

1.2、函数的使用

  
   s t r l e n strlen strlen 函数的使用非常简单,这里就不过多介绍了,大家直接看代码

#include<string.h>
#include<stdio.h>

int main()
{
	char* a = "abcdef";
	int sz = strlen(a);
	printf("%d\n", sz);
	return 0;
}


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

1.3、函数的模拟实现

(1)计数法

  首先,我们要知道指针存储的是字符串首字符的地址。该方法的思路是将传入的字符指针不断往后移每移一次计数器加一,当指针指向 ‘\0’ 时,结束计数,并返回计数的值。
  
代码实现

#include<assert.h>

size_t my_strlen(const char* str)
{
	//assert断言,防止传入空指针
	assert(str);
	int count = 0;
	
	//判断条件:指针指向的字符是否为0('\0')
	while (*str)
	{
		count++;
		str++;
	}
	return count;
}

(2)递归法

  
  有没有办法在求出字符串长度的同时不创建临时变量呢?我们可以试试递归的思想

  那用递归的思想来计算字符串的长度,解题思路是什么呢?

  我们不妨想:比如要计算 a b c abc abc 的长度,是不是可以将他看成 1 + 1+ 1+ b c bc bc 的长度,而 b c bc bc 的长度,又可以将他看成是 1+“ c c c 的长度。
  

在这里插入图片描述

  怎么样,你想到了吗?
  
代码实现

#include<assert.h>
size_t my_strlen(const char* str)
{
	assert(str);
	
	if (!*str)//逻辑取反,当*str为0时为假,逻辑取反为真
		return 0;
		
	else
		return 1 + my_strlen(str + 1);
}

  

(3)指针 - 指针

  
  还有一个方法就是指针 - 指针,首先我们知道,指针 - 指针,代表的是两指针之间的元素个数(详情请看:【C语言】—— 指针一 : 初识指针(上))
  
  我们可以利用这个思想,创建一个指针变量,让他不断移动,直至指向 '\0' ,再通过两个指针相减,得到字符串的长度。
  
代码实现

#include<assert.h>

size_t my_strlen(const char* str)
{
	assert(str);
	char* p = str;

	while (*p++)
	{
		;
	}

	return p - str - 1;
}

代码解析:
  
  while (*arr++);可能这句代码有些小伙伴不太了解,没关系,我们一起来看看

  • *arr++:这里有两个操作符: ∗ * 后置++。 虽然后置++ 的优先级比 ∗ * 高,但是他是后置的,所以先进行解引用
  • 解引用完成后,结果为真进入循环,为假退出循环
  • 无论 * a r r arr arr 结果为真为假,接下来 a r r arr arr 进行自增操作。注:++ 操作数是 a r r arr arr,而非 * a r r arr arr,如果想让 * a r r arr arr 自增,应写为:*(arr)++
  • 因为循环不需要再执行任何操作,因此放置空语句
  • return arr - str - 1;因为前面循环的判断条件不论真假, a r r arr arr 都会自增,即指针指向 ‘\0’ 后依然会往后移动一位,此时如果直接arr - str,则把 ‘\0’ 也计算在内,而 s t r l e n strlen strlen 是不计算
    ‘\0’ 的,因此最终要 - 1

  

二、 s t r c p y strcpy strcpy 函数

2.1、函数功能

在这里插入图片描述

功能 s t r c p y strcpy strcpy 函数实现的是字符串的拷贝,将 s o u r c e source source 指针指向的字符串(源字符串) 拷贝到 d e s t i n a t i o n destination destination 指针指向的字符串(目标字符串) 中。

  • 源字符串必须以 ‘\0’ 结束
  • 该函数会将 ‘\0’ 拷贝到目标空间
  • 目标空间必须足够大,以确保能存放源字符串,否则会出现越界访问的情况
  • 目标空间必须是可修改
  • 函数的返回值为指向目标空间的指针

  
  可能会有小伙伴说,字符串的拷贝哪有那么麻烦,还要用函数,不就是直接赋值吗?

int main()
{
	char arr1[] = "abcdef";
	char arr2[20] = { 0 };
	arr2 = arr1;
	return 0;
}

  这样可以吗?当然是不可以的。为什么呢?
  
  别忘了 a r r 1 arr1 arr1 a r r 2 arr2 arr2 是两个数组首元素的地址,地址是一个常量值并不是变量,就像:5 = 3;,可以把 3 赋值给 5 吗?肯定是不行的。因此字符串的赋值并没有那么简单哦。
  

2.2、函数的使用

  

#include<stdio.h>
#include<string.h>

int main()
{
	char src[20] = "good good study";
	char dest[40] = { 0 };
	char* s = strcpy(dest, src);
	printf("%s\n", s);
	return 0;
}

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

2.3、函数的模拟实现

  要模拟实现 s t r c p y strcpy strcpy 函数,其实思路很简单,只需将字符串的中的每个字符依次拷贝过去即可,当拷贝完源字符串的 '\0'拷贝结束
  
  我们来尝试写下代码:

char* my_strcpy(char* dest, const char* src)
{
    //要返回目标空间的地址
	char* ret = dest;
	
	//不能传入空指针
	assert(dest != NULL);
	assert(src != NULL);
	
	//拷贝'\0'之前的内容
	while (*src != '\0')
	{
		*dest = *src;
		dest++;
		src++;
	}
	//拷贝'\0'
	*dest = *src;
	return ret;
}

  
  上述代码基本上是没问题的,但大家发现没有,该代码有点繁琐*dest = *src;就出现了两次,有什么改进方法呢?我们来看下面的代码:
  

char* my_strcpy(char* dest, const char* src)
{
	char* ret = dest;
	assert(dest && src);

	while (*dest++ = *src++)
	{
		;
	}

	return ret;
}

  
这里,我们重点来讲解一下while (*dest++ = *src++);语句

  • 虽然两个 后置++ 的优先级最高,但是因为是后置:先使用,后++
  • 首先执行的是两个解引用操作
  • 解引用完成后,将 * s r c src src 的值赋给 * d e s t dest dest
  • 赋值完成后进行循环的条件判断,* d e s t dest dest == ‘\0’ 则退出循环,否则执行循环
  • 最后 d s e t dset dset s r c src src 两个指针执行++,即自增
      

怎么样,这段代码是不是很巧妙
  
  

三、 s t r c n t strcnt strcnt 函数

3.1、函数功能

在这里插入图片描述

功能 s t r c a t strcat strcat 函数的功能是实现字符创的追加,将源字符串的内容追加到目标字符串的结尾。

  • 源字符串必须以 ‘\0’ 结束
  • 追加会将目标空间原来的 ‘\0’ 覆盖
  • 目标字符串结尾也要有 ‘\0’,否则不知从哪开始追加
  • 目标空间必须足够大,能容下追加源字符串之后的内容
  • 目标空间必须可修改
  • 返回的是目标空间的起始地址
  • 目标空间与源字符串不能重叠,即不能自己给自己追加

  

3.2、函数的使用

  

#include<stdio.h>
#include<string.h>
int main()
{
	char src[20] = "day day up!";
	char dest[40] = "good good study! ";
	char* s = strcat(dest, src);
	printf("%s\n", s);
	return 0;
}

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

3.3、函数的模拟实现

  
   s t r c a t strcat strcat 函数怎么模拟实现呢?大家不妨想一想,追加不也是一种拷贝吗,它的实现方法与 s t r c p y strcpy strcpy 函数的实现很像,只是 s t r c p y strcpy strcpy从头开始拷贝,而 s t r c a t strcat strcat从目标字符串的末尾开始拷贝。
  
参考代码

char* mu_strcat(char* dest, const char* src)
{
	char* ret = dest;
	assert(dest && src);
    
    //找到目标空间的末尾
	while (*dest)
	{
		dest++;
	}
	
    //进行追加(拷贝)
	while (*dest++ = *src++)
	{
		;
	}
	return ret;
}

  
  

四、 s t r c m p strcmp strcmp 函数

4.1、函数功能

在这里插入图片描述

功能比较两个字符串的大小
    s t r 1 str1 str1 > s t r 2 str2 str2 ,返回 > 0 >0 >0 的数字
    s t r 1 str1 str1 = s t r 2 str2 str2 ,返回 0 0 0
    s t r 1 str1 str1 < s t r 2 str2 str2 ,返回 < 0 <0 <0 的数字
  
  
比较方式
  
   s t r c m p strcmp strcmp 函数比较字符串中对应位置上两个字符的 ASCII码 值的大小来比较两个字符串大小

   s t r c m p strcmp strcmp逐一比较两个字符串中相应位置的各个字符,从第一个字符开始比较,若相等,则比较下位字符
  直至比出大小遇到其中一个字符串的 '\0',比较结束,先遇到'\0'的字符串较小,可以理解成 ‘\0’ 的ASCII码值为 0,任何字符的 ASCII码值都比它大。
  

4.2、函数的使用

#inlcude<stdio.h>
int main()
{
	char str1[] = "abcd";
	char str2[] = "Abcd";
	char str3[] = "abcd";
	char str4[] = "bbcd";
	int ret1 = strcmp(str1, str2);//比较str1与str2
	int ret2 = strcmp(str1, str3);//比较str1与str3
	int ret3 = strcmp(str1, str4);//比较str1与str4
	printf("%d %d %d\n", ret1, ret2, ret3);
	return 0;
}

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

  

4.3、函数的模拟实现

int my_strcmp(const char* str1, const char* str2)
{
    //接下来要对指针进行解引用,因此不能传入空指针
	assert(str1 && str2);
    
    //相同位上两个字符相等,进入循环
	while (*str1 == *str2)
	{
	
	    //当两个字符都为'\0'时,也进入了循环,遇到'\0'表示字符串结束
	    //因为相等才能进入循环,判断语句因此只用写一个
		if (*str1 == '\0')
		{
			return 0;
		}
		str1++;
        str2++;
	}
	
    //不相等,返回两个字符的ASCII码的差值
	return *str1 - *str2;
}

  
  
  
  


  好啦,本期关于部分字符串函数的介绍及模拟实现就介绍到这里啦,希望本期博客能对你有所帮助,更多关于字符串的函数还请收看下一期。同时,如果有错误的地方请多多指正,让我们在C语言的学习路上一起进步!

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

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

相关文章

复习Javase 方法

Q 1: long double float blooean 这四个都不能作为switch参数类型 Q2: 数字9出现的次数 9 19 29 |....90 91....99 前面都是个位是9&#xff0c;一共是9个9、后面十位是9&#xff0c;一共11个9 问&#xff1a;如何求个位和十位的9 如果要得到个位的9 就取余 9%1…

OpenHarmony C/C++三方库移植适配

简介 众所周知&#xff0c;C/C三方库相对与JS/ETS的三方组件来说&#xff0c;其运行效率高。那如何将一个C/C三方库移植到OH系统上呢&#xff1f;本文将介绍如何快速高效的移植一个C/C三方库到OpenHarmony上。 C/C三方库适配问题与解决方案 由上图可以看出&#xff0c;三方库…

【8086汇编】汇编语言基础入门

文章目录 一、汇编简介1. 汇编语言的组成2. CPU、寄存器、内存3. CPU对存储器的读写4. 拓展5. 检测6. 解析 二、寄存器1. mov、add命令2. 物理地址3. CS:IP 装段地址和偏移地址3.1 如何改变CS:IP的值 4. 数据段DS:[address]4.1 前置知识&#xff1a;字与字节4.2 DS:[address] 5…

求存款本息和(C语言)

一、运行结果&#xff1b; 二、源代码&#xff1b; # define _CRT_SECURE_NO_WARNINGS # include <stdio.h> # include <math.h>int main() {//初始化变量值&#xff1b;double P 1000, r1 0.015, r2 0.021, r3 0.0275, r4 0.03, r5 0.0035;int judge 0;//…

MathType2024中文永久免费激活许可证

MathType是一款专业的数学公式编辑器&#xff0c;由Design Science公司开发。它为用户提供了一个便捷的方式来创建和编辑复杂的数学公式&#xff0c;适用于各种文档和出版物。MathType与常见的文字处理软件和演示程序配合使用&#xff0c;能够在各种文档中加入复杂的数学公式&a…

给自己的机器人部件安装单目摄像头并实现gazebo仿真功能

手术执行器添加摄像头 手术执行器文件夹surgical_new内容展示如何添加单目摄像头下载现成的机器人环境文件启动仿真环境 手术执行器文件夹surgical_new内容展示 进入src文件夹下选择进入vision_obliquity文件夹 选择launch 有两个可用gazebo中rviz展示的launch文件&#xff0…

基于 net/http 抽象出 go 服务优雅停止的一般思路

和其他语言相比&#xff0c;Go 中有相同也有不同&#xff0c;相同的是实现思路上和其他语言没啥差异&#xff0c;不同在于 Go 采用的是 goroutine channel 的并发模型&#xff0c;与传统的进程线程相比&#xff0c;实现细节上存在差异。 本文将从实际场景和它的一般实现方式展…

【LeetCode: 705. 设计哈希集合 + 数据结构设计】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…

【数据分析】嫡权法EWM

总结&#xff1a;基于熵值信息来计算出权重&#xff0c;数据具有客观性。 目录 简介 计算步骤 案例 简介 熵值法原理 熵值法的基本思路是根据指标变异性的大小来确定客观权重信息熵:信息量的期望。可以理解成不确定性的大小&#xff0c;不确定性越大&#xff0c;信息熵也就…

【Liunx】什么是vim?五大模式及转换方法详解

&#x1f490; &#x1f338; &#x1f337; &#x1f340; &#x1f339; &#x1f33b; &#x1f33a; &#x1f341; &#x1f343; &#x1f342; &#x1f33f; &#x1f344;&#x1f35d; &#x1f35b; &#x1f364; &#x1f4c3;个人主页 &#xff1a;阿然成长日记 …

基于SSM的在线学习系统的设计与实现(论文+源码)_kaic

基于SSM的在线学习系统的设计与实现 摘要 随着信息互联网购物的飞速发展&#xff0c;一般企业都去创建属于自己的管理系统。本文介绍了在线学习系统的开发全过程。通过分析企业对于在线学习系统的需求&#xff0c;创建了一个计算机管理在线学习系统的方案。文章介绍了在线学习系…

程序员会营销,好比虎生双翅,不是牛叉,是牛叉大发了。

Hi&#xff0c;我是贝格前端工场&#xff0c;一般来讲程序员在语言表达和营销上都是弱项&#xff0c;你看头条上那些程序员XXX&#xff0c;嘚啵嘚的能说的&#xff0c;其实都是伪程序&#xff0c;都是大商务。 不过&#xff0c;如果程序员如果能够提升自己的营销能力&#xff0…

教你将配置好的conda环境迁移到其它设备

文章目录 问题分析存在的方法环境要求方法步骤1. 下载conda pack2. 打包原环境3. 新设备还原环境4. 查看环境 问题分析 好不容易配置好的conda环境&#xff0c;要在另一个设备上运行&#xff0c;还要重新配置&#xff0c;好麻烦。 存在的方法 pip install -r requirement.txt …

CMD 汉字乱码处理

windows 11 cmd汉字乱码问题处理 一 查看CMD编码 win R 输入 cmd 输入 chcp 查看回显信息 “936”代表的意思就是 GBK (汉字内码扩展规范)&#xff0c;通常情况下GBK也是cmd的默认编码。 解决乱码需要把编码改为 utf-8 二 临时修改 在 终端中输入 chcp 65001 三 永久修改…

踩了一堆坑,终于掌握了postgreSQL主从流的精髓

&#x1f4e2;&#x1f4e2;&#x1f4e2;&#x1f4e3;&#x1f4e3;&#x1f4e3; 哈喽&#xff01;大家好&#xff0c;我是【IT邦德】&#xff0c;江湖人称jeames007&#xff0c;10余年DBA及大数据工作经验 一位上进心十足的【大数据领域博主】&#xff01;&#x1f61c;&am…

大模型化身数据魔法师,降低NLP高置信误判

关注公众号【AI论文解读】回复: 论文解读 获取本文论文 引言&#xff1a;NLP模型的高置信错误与脆弱性问题 在自然语言处理&#xff08;NLP&#xff09;领域&#xff0c;模型的预测性能优化往往伴随着高置信错误&#xff08;high confidence errors&#xff09;的产生&#x…

【python】python汽车之家数据抓取分析可视化(代码+报告+数据)【独一无二】

&#x1f449;博__主&#x1f448;&#xff1a;米码收割机 &#x1f449;技__能&#x1f448;&#xff1a;C/Python语言 &#x1f449;公众号&#x1f448;&#xff1a;测试开发自动化【获取源码商业合作】 &#x1f449;荣__誉&#x1f448;&#xff1a;阿里云博客专家博主、5…

2D AI交互数字人:赋能文旅、金融、政务、教育行业数字化转型

AI交互数字人结合了语音合成、语音识别、语义理解、图像处理、机器翻译、虚拟形象驱动等多项AI核心技术&#xff0c;可以提供服务导览、业务咨询、语音互动交流、信息播报等智能服务。 其中&#xff0c;2D AI交互数字人是采集真人视频&#xff0c;通过AI训练&#xff0c;生成逼…

C语言——字符函数与字符串函数

正文开始&#xff1a;在编程过程中&#xff0c;我们经常要处理字符和字符串&#xff0c;为了方便操作字符和字符串&#xff0c;C语⾔标准库中提供了 一系列库函数&#xff0c;接下来我们就学习⼀下这些函数。 1. 字符分类函数 C语⾔中有⼀系列的函数是专门做字符分类的&#…

基于ssm的智慧餐厅点餐管理系统设计与实现(java项目+文档+元)

风定落花生&#xff0c;歌声逐流水&#xff0c;大家好我是风歌&#xff0c;混迹在java圈的辛苦码农。今天要和大家聊的是一款基于ssm的智慧餐厅点餐管理系统。项目源码以及部署相关请联系风歌&#xff0c;文末附上联系信息 。 项目简介&#xff1a; 智慧餐厅点餐管理系统设计…