详解C语言string.h中常用的14个库函数(三)

news2025/1/13 2:51:17

string.h中的库函数
本篇博客继续讲解C语言string.h头文件中的库函数。本篇博客计划讲解3个函数,分别是:strstr, strtok, strerror。其中strstr函数我会用一种最简单的方式模拟实现。

strstr

char * strstr ( const char * str1, const char * str2 );

strstr可以在str1中查找str2第一次出现的位置,并且返回一个指针指向该位置。如果找不到,就返回NULL指针。

比如:

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

int main()
{
	char arr1[] = "abcccccdecccdefg";
	char arr2[] = "cccd";
	char* ret = strstr(arr1, arr2);
	if (ret == NULL)
		printf("找不到\n");
	else
		printf("%s\n", ret);

	return 0;
}

输出结果:
在这里插入图片描述
由于返回了cccd第一次出现的位置,顺着该地址往后打印,就有了以上的结果。

这个函数的使用是非常简单的,它的难点在于如何实现。有一些比较复杂的算法,比如KMP算法,可以实现类似的功能,但是难度较大,不适合初学者,这里我使用一种暴力查找的思路来实现。

首先,搭出框架:

char* my_strstr(const char* str1, const char* str2)
{
	assert(str1 && str2);

	const char* cp = str1;
	
	while (*cp)
	{
		cp++;
	}
}

我的想法是,先用cp指针,指向str1的位置,然后使用该指针向后遍历str1这个字符串,看看能不能找到str2,如果找到了就返回cp指针。那每次如何查找呢?可以再定义2个指针s1和s2,s1从cp的位置向后遍历,s2从str2的位置向后遍历。

char* my_strstr(const char* str1, const char* str2)
{
	assert(str1 && str2);

	const char* cp = str1;
	const char* s1 = str1;
	const char* s2 = str2;

	while (*cp)
	{
		s1 = cp;
		s2 = str2;

		cp++;
	}
}

s1和s2指向的字符,如果相等,就遍历下一对,如果s2遇到了\0,就说明找到了。

char* my_strstr(const char* str1, const char* str2)
{
	assert(str1 && str2);

	const char* cp = str1;
	const char* s1 = str1;
	const char* s2 = str2;

	while (*cp)
	{
		s1 = cp;
		s2 = str2;
		while (*s1 && *s2 && *s1 == *s2)
		{
			++s1;
			++s2;
		}
		if (*s2 == '\0')
		{
			return (char*)cp;
		}

		cp++;
	}

	return NULL;
}

strtok

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

strtok这个函数较为复杂,想要理解并且灵活使用,需要理解以下几点:

  1. strtok是用来分割字符串的。有些时候,我们想根据某些分隔符来分割一个字符串,就可以使用strtok。比如:"fish@qq.com",如果我们想根据@.这2个分隔符来分割该字符串,就可以分割成3个字符串,分别是:fish, qq, com
  2. strtok的第1个参数是待分割的字符串。第2个参数是分隔符的集合,即分隔符(一些字符)组合成的字符串。比如,如果我们想用@.来分割fish@qq.com这个字符串(假设把这个字符串存储到了字符数组arr中char arr[] = "fish@qq.com";),可以这么写:char* ret = strtok(arr, "@.");。其中"@."是由1个或多个分隔符组成的字符串。
  3. 如果“正常”调用该函数,即使用第2点的方式,strtok函数会找到字符串中的第1个分隔符出现的地方,把这个分隔符改成\0,并且返回由该分隔符分割的字符串(这个字符串在分隔符前面)。这么说有点抽象,以第2点的例子为例,strtok会在arr中找到第一个@,把@改成\0,此时arr数组存储的就是:"fish\0qq.com",函数会返回f的地址,此时如果使用printf以%s的格式打印这个返回的地址printf("%s\n", ret);,就会打印出fish。
  4. 如果调用该函数时,第一个参数为NULL指针,函数会从同一个字符串上次查找到的分隔符的位置开始向后找,找到下一个分隔符,然后返回以该分隔符分割的字符串(这个字符串在分隔符前面)。比如,假设已经像第2、3点所示调用过一次strtok函数了,如果再这么调用:ret = strtok(NULL, "@.");,函数就会找到.所在的位置,并把.改成\0,此时arr中存储的就是:"fish\0qq\0com",函数会返回第一个q的地址,此时以同样的方式printf("%s\n", ret);打印这个返回值,就会打印qq。同理,再调用一次ret = strtok(NULL, "@.");,然后再打印printf("%s\n", ret);就会打印出com。
  5. 如果函数没有找到下一个标记,就会返回NULL指针。

所以,根据以上知识点,如果我想要分割一个字符串"hello@world.nihao-shijie@this+is@a.test",可以使用一个循环来搞定。注意,一般来说,我们都会先把这个字符串拷贝一份,再来做分割,因为strtok函数会改变原字符串。

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

int main()
{
	char arr[] = "hello@world.nihao-shijie@this+is@a.test";
	char bak[100] = { 0 };
	strcpy(bak, arr);
	const char* sep = "@.+-";
	char* ret = strtok(arr, sep);
	while (ret)
	{
		printf("%s\n", ret);
		ret = strtok(NULL, sep);
	}

	return 0;
}

只有第一次调用是“正常”调用,其他每次调用第一个参数都传NULL指针,这样就会从上次找到的位置接着向后找。哪次没有找到就会返回NULL指针,跳出循环。

输出结果:
在这里插入图片描述

strerror

char * strerror ( int errnum );

strerror函数会返回错误码对应的错误信息。使用起来非常简单,比如我强行制造一个错误,让malloc函数开辟很大一块空间。当然,strerror智能检测库函数调用时的错误。

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

int main()
{
	size_t n = -1; // 制造一个很大的数
	int* p = (int*)malloc(n);
	if (p == NULL)
	{
		printf("malloc: %s\n", strerror(errno));
		return 1;
	}
	// ...
	free(p);
	p = NULL;

	return 0;
}

在这里插入图片描述

可以发现,malloc函数开辟空间失败时,使用strerror,把错误码errno转换成了一个字符串,并且返回这个字符串的起始位置。顺着这个位置打印,就打印出了错误信息:“Not enough space”。

errno是一个全局变量,如果库函数调用失败,就会设置错误码,而strerror就可以把该错误码转换成错误信息。注意,errno的使用需要引用头文件errno.h。

当然,如果只想打印错误信息,可以直接使用perror函数,比如以上程序就可以把printf("malloc: %s\n", strerror(errno));换成perror("malloc");。各位可以自行验证,以上2句代码是完全等价的。

总结

  1. strstr函数可以在一个字符串中查找子串,如果找到,就返回第一次出现的位置;如果找不到返回NULL指针。
  2. strtok函数可以分割一个字符串。如果“正常”调用,即第一个参数不为NULL,就会去找分隔符第一次出现的位置;如果第一个参数为NULL,就会从上一次找到的位置开始向后找。
  3. strerror可以把错误码转换成错误信息。当库函数调用出现错误时,编译器会把错误码保存到errno这个全局变量中,而strerror可以把errno对应的错误码转换成错误信息。

感谢大家的阅读!

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

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

相关文章

电脑怎么远程控制另一台电脑

要从一台电脑远程控制另一台电脑&#xff0c;您可以使用远程桌面软件。 以下是远程控制另一台电脑的步骤&#xff1a; 一、在两台电脑上安装远程桌面软件 有多种远程桌面软件可用&#xff0c;例如 Splashtop、微软远程桌面。 在远程电脑和本地电脑上分别安装软件。访问各自软…

MQ集群(rabbitMQ)

普通集群: 创建过程 我们先在之前启动的mq容器中获取一个cookie值&#xff0c;作为集群的cookie。执行下面的命令&#xff1a; docker exec -it mq cat /var/lib/rabbitmq/.erlang.cookie 可以看到cookie值如下&#xff1a; EFLYOBKNYFETSAXOUEYI 接下来&#xff0c;停止…

中级软件设计师备考---操作系统和计算机网络

【因为我自己是软件工程专业毕业的学生&#xff0c;所以408里的这两门课都比较熟悉&#xff0c;因此这一部分只放一些我印象不是完全深刻的知识。】 目录 操作系统前驱图与PV操作死锁的预防与避免绝对路径和相对路径缺页中断的某种练习题 计算机网络网络规划与设计特殊含义的I…

Jenkins Kubernetes

Kubernetes集成Harbor Harbor 私服配置 在Kubernetes的master和所有worker节点上加上harbor配置&#xff0c;修改daemon.json&#xff0c;支持Docker仓库&#xff0c;并重启Docker。 sudo vim /etc/docker/daemon.json {"registry-mirrors": ["https://jrabv…

Talk预告 | 北京大学刘博UCL冯熙栋-TorchOpt,一个高效的可微优化库

本期为TechBeat人工智能社区第490期线上Talk&#xff01; 北京时间4月13日(周四)20:00&#xff0c;北京大学智能科学与技术研究助理—刘博和伦敦大学学院 &#xff08;UCL&#xff09; 计算机博士生—冯熙栋的Talk将准时在TechBeat人工智能社区开播&#xff01; 他们与大家分…

【具体到每一步】从0制作一个uniapp的新闻类页面(界面篇)

目录 项目初始化 / 基础配置 项目创建 配置路由/页面/tabbar pages.json配置tabbar 配置图标/静态资源 导航栏和字体颜色 scroll-view实现横向滚动条样式 公共模块定义components组件 新建组件 使用组件 组件里的结构 布局个人中心页面 组件差异化处理 数据传递 导航…

【角色用例图讲解】

角色用例图讲解 一、 用例图的定义及组成要素二、参与者与用例的4种关系三、 常用的用例图软件 我是将军我一直都在&#xff0c;。&#xff01; 一、 用例图的定义及组成要素 用例图是编写需求说明时经常用到的需求表达方式&#xff0c;用于向开发、测试同事说明需求中用户与系…

【机智云物联网低功耗转接板】+模拟无线远程控制仪表启停

GE211 是机智云自研的定制化的物联网转接板&#xff0c;使用 ESP32-C3-WROOM-02 通讯模块&#xff0c;适用于白色智能家电等设备应用。硬件设计上采用支持 WiFi 和 BLE 的双模无线通信模块&#xff0c;具备TTL电平转换串口&#xff0c;免开发快速接入&#xff0c;提高开发效率。…

SpringBoot 使用 Sa-Token 完成权限认证

一、设计思路 所谓权限认证&#xff0c;核心逻辑就是判断一个账号是否拥有指定权限&#xff1a; 有&#xff0c;就让你通过。没有&#xff1f;那么禁止访问&#xff01; 深入到底层数据中&#xff0c;就是每个账号都会拥有一个权限码集合&#xff0c;框架来校验这个集合中是…

Linux 常用命令2

文件内容查看 Linux系统中使用以下命令来查看文件的内容&#xff1a; cat 由第一行开始显示文件内容tac 从最后一行开始显示nl 显示的时候&#xff0c;顺道输出行号more 一页一页的显示文件内容less与more 类似&#xff0c;但可以往前翻页head 只看头几行tail 只看尾巴几行 j…

【ZooKeeper to KRaft Migration】kafka 3.4版本zookeeper数据迁移到kraft

引言 kafka在3.X版本后内置了kraft用来替代zookeeper管理集群&#xff0c;但是在升级的过程中发现&#xff0c;许多升级的文档都是只有新部署安装kraft版本&#xff0c;而没有涉及到数据迁移相关的资料&#xff0c;这样如果直接变更的话&#xff0c;会导致kakfa中的数据全部丢…

升级gpt4-GPT最新版本怎么下载使用

怎么 让gpt-3的模型升级gpt4 GPT-4是OpenAI的未来版本&#xff0c;目前还未发布。因此&#xff0c;我们无法准确指导如何将GPT-3升级到GPT-4。要升级GPT-3&#xff0c;需要进行大量的研究和开发工作。如果OpenAI发行了GPT-4的预览版&#xff0c;那么可能需要花费大量的时间和资…

Redission实现分布式锁lock()和tryLock()方法的区别

lock.lock(30, TimeUnit.SECONDS); // 尝试获取锁30秒&#xff0c;如果获取不到则放弃 //尝试获取锁&#xff0c;等待5秒&#xff0c;持有锁10秒钟 boolean success lock.tryLock(0, 10, TimeUnit.SECONDS); Redisson 是一种基于 Redis 的分布式锁框架&#xff0c;提供了 lo…

国产蓝牙芯片OM6621P/HS6621系列门锁方案

在5G、物联网以及互联网家装市场的快速发展等多重因素的作用下&#xff0c;中国智能家居市场展现蓬勃发展态势。作为智能家居“入口”产品以及家庭智能安防产品的核心单品&#xff0c;智能门锁以其区别于传统机械锁更具安全性、便利性、可扩展性的优势&#xff0c;逐渐成为智能…

Java+GeoTools(开源的Java GIS工具包)快速入门-实现读取shp文件并显示

场景 GeoTools GeoTools 是一个开源的 Java GIS 工具包&#xff0c;可利用它来开发符合标准的地理信息系统。 GeoTools 提供了 OGC (Open Geospatial Consortium) 规范的一个实现来作为他们的开发。 官网地址: GeoTools The Open Source Java GIS Toolkit — GeoTools 参考…

Linux账号密码安全策略设置

前言 随着云计算厂商的兴起&#xff0c;云资源如ECS不再只有企业或者公司才会使用&#xff0c;普通人也可以自己买一台ECS来搭建自己的应用或者网站。虽然云计算厂商帮我们做了很多安全相关的工作&#xff0c;但并不代表我们的机器资源就绝对是安全的。 要知道有很多事情是云…

群策群力:组织效率,管理?沟通?协作?

你好&#xff0c;我是苏杰。今天让我们一起聊聊组织效率的话题。 团队大了&#xff0c;也能够自我造血以后&#xff0c;如何可持续发展就会成为我们关注的焦点。产品会衰退、行业有生命周期&#xff0c;但人的成长&#xff0c;以及人构成的组织&#xff0c;可以帮我们不断成功…

【大厂直通车】飞猪旅行日常实习_测开面经

📑哈喽,大家好,我是小浪;📱本专栏致力于持续更新最新各大厂面经,实习消息,招聘要求; 那么目前价格也仅仅是定到了29.9💰;非常的实惠,一杯奶茶钱🍵; 🧃对于订阅本专栏的同学们,博主在努力更新,那么最近忙于学校的考试,没来得及正常更新,非常抱歉,这几…

论文综述——DORE: Document Ordered Relation Extraction based on Generative Framework

DORE: Document Ordered Relation Extraction based on Generative Framework 文章的主要目标是对文档级的关系抽取。以往的研究主要是基于分类的研究&#xff0c;生成式关系抽取研究较少而且性能不佳。 文档级相比于句子级的关系抽取存在序列长度过长&#xff0c;以及实体定位…

消息队列选型

消息队列选型 大家好&#xff0c;我是易安&#xff01;今天我们聊下消息队列常见选型。 消息队列作用 谈选型之前我们先讲下我们为什么需要消息队列。 消息队列是一种很流行的技术&#xff0c;自从系统间开始通信时&#xff0c;消息队列就出现了。然而&#xff0c;对消息队列给…