字符串函数(二):strlen(求长度),strstr(查找子串),strtok(分割),strerror(打印错误信息)

news2025/1/16 2:48:49

字符串函数

  • 一.strlen(求字符串长度)
    • 1.函数使用
    • 2.模拟实现(三种方法)
  • 二.strstr(字符串查找子串)
    • 1.函数使用
    • 2.模拟实现
  • 三.strtok(字符串分割)
  • 四.strerror,perror(打印错误信息)

一.strlen(求字符串长度)

1.函数使用

size_t strlen(const char* str);
  • strlen函数用于求字符串的长度,参数是字符串的首地址,返回值是无符号的整形,初始化字符串有两种,它们各自存在一些小的细节,如下:
#include<stdio.h>
#include<string.h>
int main()
{
	char str1[] = "abcdef";//本质:{ 'a','b','c','d','e','f','\0' };隐藏了\0在末尾
	char str2[] = { 'a','b','c','d','e','f' };

	//函数strlen(): 求字符串长度
	printf("%zu\n", strlen(str1));//6
	printf("%zu\n", strlen(str2));//由于尾部没有'\0'随机数,所以打印随机数
	printf("%s\n", str2);//直到找到'\0'为止,停止打印

	//关键字sizeof():求字节的大小返回值同样是size_t(unsigned int)
	printf("%zu\n", sizeof(str1));//7
	printf("%zu\n", sizeof(str2));//6
	return 0;
}

在这里插入图片描述

  • 可以看到str2的长度居然是33,其实这是因为strlen函数会从首地址指向的字符一直向后查找,直到遇到’\0’,才会停下(不包含 ‘\0’ ),统计’\0’之前出现的字符的个数,而str2末尾没有’\0’,所以会一直向后查找,在某个位置恰好遇到了’\0’,所以打印随机数33。
  • printf函数打印字符串也是同样的道理,看似传入"abcdef",其实真正传入的是首地址(a的地址),遇到了’\0’,停止打印。那为什么会出现烫烫烫呢?这就牵扯到了函数栈帧了,日后会将函数栈帧更新上来。

总结:

  • 字符串以 ‘\0’ 作为结束标志,strlen函数返回的是在字符串中 ‘\0’ 前⾯出现的字符个数(不包
    含 ‘\0’ )。

  • 参数指向的字符串必须要以 ‘\0’ 结束。

  • 注意函数的返回值为size_t,是无符号整形(size_t 等价于 unsigned int)。

易错点:

#include<stdio.h>
int main()
{
	if (strlen("abc") - strlen("abcdef"))
		printf(">\n");
	else
		printf("<\n");
	return 0;
}

在这里插入图片描述

  • 我们会发现输出的居然大于号,这是因为strlen函数的返回值是无符号的整形,而无符号的整形相减仍然得到无符号的整形,而-3按照无符号整形来说是一个很大的数,自然打印大于号。

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

  • 也可以这么写:
#include<stdio.h>
int main()
{
    //if ((int)strlen("abc") - (int)strlen("abcdef") >0)
	if (strlen("abc") > strlen("abcdef"))
		printf(">\n");
	else
		printf("<\n");
	return 0;
}

2.模拟实现(三种方法)

  • 方法一:计数器方法
  • 我们定义一个变量为count,如果传入的指针指向的内容不是’\0’,那么count++,同时指针后移一位,循环往复,直到找到’\0’时返回count即可。
#include<assert.h>
size_t my_strlen(const char* str)
{
	assert(str != NULL);//断言,若str为NULL,报错,头文件assert.h
	size_t count = 0;
	while (*str != '\0')
	{
		str++;
		count++;
	}
	return count;
}
  • 方法二:指针减指针
  • 我们先定义两个指针变量start与end将传入的指针保存下来,然后将指针end向后移,直到遇到’\0’时,我们返回指针end与指针start的差值即可。(指针与指针的差的绝对值是两个指针之间的元素个数)。
#include<assert.h>
size_t my_strlen(const char* str)
{
	assert(str != NULL);//断言,若str为NULL,报错,头文件assert.h
	const char* start = str;
	const char* end = str;
	while (*end != '\0')
	{
		end++;
	}
	return end - start;
}
  • 方法三:递归
  • 判断传入指针指向的内容是否为’\0’,如果是就返回0,不是就返回my_strlen2(str+1)+1,如此进行下去,直到递归到内层时找到’\0’,这时再一步步将值返回回来即可。
#include<assert.h>
size_t my_strlen(const char* str)
{
    assert(str != NULL);//断言,若str为NULL,报错,头文件assert.h
	if (*str == '\0')
		return 0;
	else
		return my_strlen(str + 1) + 1;
}

二.strstr(字符串查找子串)

1.函数使用

char* strstr(const char* str1, const char* str2)
  • strstr函数用于在字符串 str1 中查找另一个字符串 str2,如果字符串 str1 存在字符串 str2,那么就返回字符串 str2 在字符串 str1 中第一次出现的起始位置,如果找不到那么就返回空指针(NULL)。它的第一个参数是字符串 str1 的首地址,第二个参数是字符串 str2 的首地址。
  • 若字符串 str2 为空字符串,则返回字符串 str1 的首地址。
#include<stdio.h>
#include<string.h>
int main()
{
	char arr1[] = "ABCDEFG";
	char arr2[] = "CDE";
	char* ret = strstr(arr1, arr2);//返回arr1中字符C的地址
	if (ret != NULL)
		printf("%s\n", ret);//打印CDEFG
	else
		printf("子串不存在\n");
	return 0;
}

2.模拟实现

例如,在字符串"abbbcdef"中查找字符串"bbc":

  • p指针: 记录每次开始匹配时的起始位置,当从该位置开始匹配时就找到了目标字符串,便于返回指针p;当从该位置开始没有匹配成功时,则指针p后移一位进行下一次的匹配。
  • s1和s2指针: 通过判断s1和s2指针解引用后是否相等来判断单个字符是否匹配成功。若成功,则指针s1与s2后移一位比较下一对字符;若失败,指针p后移一位,指针s1返回指针p处,指针s2返回待查找字符串的起始位置。

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

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
代码实现:

#include<assert.h>
#include<stdio.h>
char* my_strstr(const char* str1, const char* str2)
{
	assert(str1 && str2);
	const char* s1 = str1;
	const char* s2 = str2;
	const char* p = str1;
	if (*str2 == '\0')//要查找的字符串为空字符串
	{
		return str1;
	}
	while(*p != '\0')//若要减少循环数则可以修改为
	//while ((p - str1) <= (int)strlen(str1) - (int)strlen(str2))
	{
		while (*s1 == *s2 && *s1 != '\0' && *s2 != '\0')
	    {
			s1++;
			s2++;
		}
		if(*s2=='\0')//字符串查找子串成功
		{
			return p;
		}
		p++;
		s1 = p;
		s2 = str2;
	}
	return NULL;//找不到目标字符串,返回NULL
}
int main()
{
	char arr1[] = "abbbcdef";
	char arr2[] = "bbc";
	char* ret = my_strstr(arr1, arr2);
	if (ret != NULL)
		printf("%s\n", ret);
	else
		printf("子串不存在\n");
	return 0;
}

对于以上代码,对于循环次数还可以优化下,比如待比较的字符串str2长度大于字符串str1还能比较的字符的个数时,是不可能查找到的,所以:

while(*p != '\0')//若要减少循环数则可以修改为
while ((p - str1) <= (int)strlen(str1) - (int)strlen(str2))

三.strtok(字符串分割)

char* strtok (char* str, const char* sep);

strtok函数能通过给定的一系列字符将一个字符串分割成许多子字符串的函数。它的第一个参数是需要被分割的字符串的首地址;第二个参数是一个字符串的首地址,该字符串是用作分隔符的字符集合。返回值是查找到的标记的首地址。

注意:

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

例如:

#include<stdio.h>
#include<string.h>
int main()
{
	char email[] = "3315898248@qq.com";//待分割字符串
	char* sep = "@.";//分隔符的字符集合
	char cp[20] = { 0 };
	strcpy(cp, email);//将数据拷贝一份使用,防止原数据被修改

	//char* ret = strtok(email, sep);//第一次传参需传入待分割字符串首地址
	//while (ret != NULL)//说明还未分割完
	//{
	//	printf("%s\n", ret);
	//	ret = strtok(NULL, sep);//对同一个字符串进行分割,第二次及以后的第一个参数为NULL
	//}

	//或者
	char* ret = NULL;
	//利用for循环的特点,先ret = strtok(email, sep);判断ret != NULL;
	//打印ret,再ret = strtok(NULL, sep);判断ret != NULL;循环到ret == NULL结束
	for (ret = strtok(email, sep); ret != NULL; ret = strtok(NULL, sep))
	{
		printf("%s\n", ret);
	}
	return 0;
}

四.strerror,perror(打印错误信息)

char* strerror (int errnum);

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

注意:

  1. 在不同的系统和C语言标准库的实现中都规定了一些errno(错误码),一般是放在 errno.h 这个头文件中说明的。
  2. C语言程序启动的时候就会使用一个全局变量errno来记录程序的当前错误码,只不过程序启动的时候errno是0,表示没有错误,当我们在使用标准库中的函数的时候发生了某种错误,就会将对应的错误码,存放在errno中。
  3. 而一个错误码的数字是整数很难理解是什么意思,所以每一个错误码都是有对应的错误信息的。strerror函数就可以将错误对应的错误信息字符串的地址返回。
#include<errno.h>
#include<string.h>
#include<stdio.h>
//我们打印一下0~10这些错误码对应的信息 
int main()
{
	int i = 0;
	for (i = 0; i <= 10; i++)
	{
		printf("%s\n", strerror(i));
	}
	return 0;
}

在这里插入图片描述
例如:打开一个文件(fopen函数表示:打开文件(读或写),当其执行成功时会返回文件的首地址,执行失败时会返回一个空指针(NULL)。)

int main()
{
	FILE* pf = fopen("test.txt", "r");//打开文件 ———— 读文件
	if (pf == NULL)//若为NULL,文件打开失败,下一步查看错误信息
	{
		printf("%s\n", strerror(errno));//将错误码转换成对应的错误信息
		perror("fopen");//直接打印错误信息
		return 1;
	}
	return 0;
}

在这里插入图片描述

  1. printf(“%s\n”, strerror(errno)) 等价于 perror(“”)
  2. perror(“fopen”):可以代替strerror(errno)且效果更好,加上(fopen)来明确错误来源于哪里。
  3. 程序执行会加上一个冒号。

创作不易,如果能帮到你的话能赏个三连吗?感谢啦!!!
在这里插入图片描述

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

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

相关文章

BLDC电机基础知识

1、电机工作原理 电机输入的是电能输出机械能&#xff0c;即电机是一种将电能转换为机械能的装置。电机利用磁场的同名磁极互相排斥以及电磁场原理完成电能与机械能的转换。 由物理电磁场理论知识我们知道&#xff0c;磁铁周围存在磁场&#xff0c;同时运动的电荷或通电导线周…

【强训笔记】day22

NO.1 思路&#xff1a;将情况全部枚举出来。 代码实现&#xff1a; #include <iostream> #include<string> using namespace std;string a,b; int main() {cin>>a>>b;int ma.size(),nb.size();int retm;for(int i0;i<n-m;i){int tmp0;for(int j…

Linux修改终端命令颜色

1.在家目录中修改.bashrc文件 cd ~ vim .bashrc2.找到PS1相关段落&#xff0c;把其他的注释掉&#xff0c;填上该行代码&#xff0c;修改为自己设置的颜色 (具体颜色查看参考文章) 提供两种颜色&#xff0c;其他的自学调色盘吧(下文有)~ (祝你愉快) ①浅蓝色 深蓝 PS1\[\03…

知识图谱 | 语义网络写入图形数据库(含jdk和neo4j的安装过程)

Hi&#xff0c;大家好&#xff0c;我是半亩花海。本文主要介绍如何使用 Neo4j 图数据库呈现语义网络&#xff0c;并通过 Python 将语义网络的数据写入数据库。具体步骤包括识别知识中的节点和关系&#xff0c;将其转化为图数据库的节点和边&#xff0c;最后通过代码实现数据的写…

pytorch学习(一):tensorboard使用

第一次使用需要加入 pip install tensorboard安装后&#xff0c;就可以使用tensorboard&#xff1b; from torch.utils.tensorboard import SummaryWriterwriterSummaryWriter("logs")# writer.add_image() for i in range(100):writer.add_scalar("yx",…

JWT令牌技术实现登录校验

一.简单登录功能 在登录界面中&#xff0c;我们可以输入用户的用户名以及密码&#xff0c;然后点击 "登录" 按钮就要请求服务器&#xff0c;服务端判断用户输入的用户名或者密码是否正确。如果正确&#xff0c;则返回成功结果&#xff0c;跳转至系统首页面。 1.功能…

LeetCode1657确定两个字符串是否接近

题目描述 如果可以使用以下操作从一个字符串得到另一个字符串&#xff0c;则认为两个字符串 接近 &#xff1a; 操作 1&#xff1a;交换任意两个 现有 字符。例如&#xff0c;abcde -> aecdb操作 2&#xff1a;将一个 现有 字符的每次出现转换为另一个 现有 字符&#xff0…

智慧公厕系统:改变“上厕所”体验的科技革新

公共厕所是城市建设中不可或缺的基础设施&#xff0c;然而&#xff0c;由于较为落后的管理模式&#xff0c;会常常存在着管理不到位、脏乱差的问题。为了改善公厕的使用体验&#xff0c;智慧公厕系统应运而生&#xff0c;并逐渐成为智慧城市建设的重要组成部分。本文将以智慧公…

6. RedHat认证-基于公钥的认证方式

6. RedHat认证-基于公钥的认证方式 主要学习客户端访问服务端的时候&#xff0c;免密登录这一方式 注意: 免密登录只是基于公钥认证的一个附带属性(基于公钥认证的方式更加安全&#xff0c;防止黑客暴力破解) 第一步&#xff1a;将客户端生成的秘钥传送到服务器 在客户端通过…

display:flex align-items:center无效的不一样的解决思路

写H5的时候&#xff0c;希望两个元素在div中垂直居中&#xff0c;但是设置align-items:center无效&#xff0c;最终排查原因是引入三方css影响了align-items:center。 具体分析如下&#xff0c;想让搜索图标和input在div里水平居中&#xff1a; 布局如下&#xff1a; <div…

实验十 智能手机互联网程序设计(微信程序方向)实验报告

实验目的和要求 完成以下页面设计。 二、实验步骤与结果&#xff08;给出对应的代码或运行结果截图&#xff09; Wxml <view class"container"> <view class"header"> <view class"logo"…

MHD093C-058-PG1-AA具备哪些特点?

MHD093C-058-PG1-AA是一种高性能的伺服电机控制器。 该产品具备以下特点&#xff1a; 高精度与高性能&#xff1a;MHD093C-058-PG1-AA设计用于提供精确的运动控制和定位&#xff0c;适用于需要高精度定位和控制的场合。快速响应&#xff1a;采用先进的控制技术&#xff0c;确…

C++类与对象基础探秘系列(二)

目录 类的6个默认成员函数 构造函数 构造函数的概念 构造函数的特性 析构函数 析构函数的概念 析构函数的特性 拷贝构造函数 拷贝构造函数的概念 拷贝构造函数的特性 赋值运算符重载 运算符重载 赋值运算符重载 const成员 const修饰类的成员函数 取地址及const取地址操作…

扫码枪与Input的火花

文章目录 前言一、需求&#xff1a;交互细节二、具体实现两个核心的函数&#xff1a;自动聚焦 三&#xff0c;扩展知识input 与 change的区别 前言 在浏览器扫描条形码获取条形的值&#xff0c;再操作对应的逻辑。这是比较常见的业务&#xff0c;这里记录实际操作。 其中PC端…

LeetCode2095删除链表的中间节点

题目描述 给你一个链表的头节点 head 。删除 链表的 中间节点 &#xff0c;并返回修改后的链表的头节点 head 。长度为 n 链表的中间节点是从头数起第 ⌊n / 2⌋ 个节点&#xff08;下标从 0 开始&#xff09;&#xff0c;其中 ⌊x⌋ 表示小于或等于 x 的最大整数。对于 n 1、…

Leaflet系列——【一】初识Leaflet与Leaflet视图操作

初识Leaflet&#xff08;vue3 &#xff09; 前言&#xff1a;当你熟悉了openlayer、mapbox、cesium等一些GIS框架之后&#xff0c;对于我们开发来说其实他们的本质就是往瓦片上面叠加图层、【点、线、面、瓦片、geoJson、热力图、图片、svg等等】都是一层层的Layer图层&#xf…

AI与人类生活的融合:安克创新CEO阳萌的深度洞见

安克创新CEO阳萌分享了他对人工智能未来发展的深刻见解。阳萌不仅深入探讨了大模型技术的应用前景&#xff0c;还对AI与人类生活的融合提出了引人入胜的思考。以下是对这次访谈内容的总结和分析。 大模型技术的现实应用 阳萌提到&#xff0c;尽管大模型在处理通用知识方面表…

拥有一个生产女朋友的工厂,是什么体验

正所谓&#xff0c;“旱的旱死&#xff0c;涝的涝死”&#xff0c;在这个充满竞争的编程界&#xff0c;我们似乎总是忙于解决bug和优化算法&#xff0c;以至于个人生活常常被忽略。但别担心&#xff0c;今天&#xff0c;我们要用一种独特的方式&#xff0c;解决这个“问题”。 …

OpenAI Whisper 语音转文本实验

为了实现语音方式与大语言模型的对话&#xff0c;需要使用语音识别&#xff08;Voice2Text&#xff09;和语音输出&#xff08;Text2Voice&#xff09;。感觉这项技术已比较成熟了&#xff0c;国内也有许多的机构开发这项技术&#xff0c;但是像寻找一个方便测试的技术居然还不…

根据后端返回下拉请求地址,前端动态请求拿到下拉数据渲染

完整代码如下&#xff1a; <template> <!-- 资源列表页 --> <div> <div> <i click"$router.go(-1)" style" color: #409eff; cursor: pointer; margin-right: 5px; font-size: 18px; " class"el-icon-back" ><…