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

news2024/11/15 13:42:06

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/447387.html

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

相关文章

用yolov5+playwright过滑动验证码

目录 梳理思路 训练模型 编写代码 总结与提高 源码下载 在上一节&#xff0c;我们通过opencv-pythonplaywright成功过掉了QQ空间的滑动验证码。在本节&#xff0c;我们将使用yolov5playwright来实现相同效果。 注&#xff1a;因为yolov5的配置教程网上已经很多了&#xff…

C++初阶之函数重载

目录 前言 函数重载 1.函数重载的概念 2.C支持函数重载的原理--名字修饰(name Mangling) 前言 今天小编给大家带来的是C中关于函数重载的内容&#xff0c;和C语言不一样&#xff0c;函数重载是C语言特有的&#xff0c;那么该功能实现的底层原理是什么呢&#xff1f;请大家…

Idea配置maven,指定settings.xml文件不生效

一.简介 最近单位要求把项目的仓库配置从阿里云改为nexus私服&#xff0c;配置了一个settings-nexus.xml的配置文件&#xff0c;idea的maven配置指定了该settings-nexus.xml文件&#xff0c;发现走的还是阿里云的&#xff0c;新的settings-nexus.xml竟然不生效&#xff0c;依赖…

分支和循环语句(1)

文章目录 目录1. 什么是语句2. 分支语句&#xff08;选择结构&#xff09;2.1 if语句2.1.1 悬空else2.1.2 if书写形式的对比2.1.3 练习 2.2 switch语句2.2.1 在switch语句中的 break2.2.2 default子句2.2.3 练习 3. 循环语句3.1 while循环3.1.1 while语句中的break和continue 附…

记一次fastjson反序列化到内网靶标

声明&#xff1a;文中涉及到的技术和工具&#xff0c;仅供学习使用&#xff0c;禁止从事任何非法活动&#xff0c;如因此造成的直接或间接损失&#xff0c;均由使用者自行承担责任。 点点关注不迷路&#xff0c;每周不定时持续分享各种干货。 众亦信安&#xff0c;中意你啊&a…

多种方法解决This is usually caused by another repository pushing to the same ref的错误

文章目录 1. 复现错误2. 分析错误3. 解决错误4. 解决该错误的其他方法 1. 复现错误 今天使用git status查看文件状态&#xff0c;发现有一个文件未提交&#xff0c;如下代码所示&#xff1a; D:\project\test>git status On branch master Your branch is up to date with …

sftp常用命令介绍

sftp常用命令&#xff1a; 1. sftp 登录sftp服务器 sftp userip ​​​​​​ 如需要看全部命令&#xff1a;则使用help即可 2. pwd和lpwd 、 ls和lls 、cd和lcd 等 sftp登录之后默认操作是远程服务器&#xff0c;当需要操作本地时&#xff0c;就需要在前边加“l”&#…

【wpf踩坑日记】搞错了,眼睛问题(:))

背景 今天遇到一个草鸡奇葩的问题&#xff1a; ComboBox 选择时 没有触发绑定的属性的set。 其实看错了&#xff0c;Mode写出OneWay&#xff0c;应该是TowWay。 不然是会触发set的。兄弟们不用往下看了。。。。。 哎&#xff0c;有的时候就会碰到这种情况&#xff0c;我还…

Ubuntu上跑通PaddleOCR

书接上文。刚才说到我已经在NUC8里灌上了Windows Server 2019。接下来也顺利的启用了Hyper-V角色并装好了一台Ubuntu 22.04 LTS 的虚机。由于自从上回在树莓派上跑通了Paddle-Lite-Demo之后想再研究一下PaddleOCR但进展不顺&#xff0c;因此决定先不折腾了&#xff0c;还是从x6…

python常见问题总结

对于长期深耕在python爬虫的程序员来说&#xff0c;如何快速解决代码中的问题它是作为合格的程序员应该具备的基本素质。下面将我总结整理出有关python的一些常见问题记录下来方便后期查证。 Python python 没有多态&#xff0c;而是鸭子类型 多继承&#xff0c;没有接口&…

【操作系统】CPU 缓存一致性

【操作系统】CPU 缓存一致性、MESI 协议 参考资料&#xff1a; CPU缓存一致性协议(MESI) 【JUC】Java并发机制的底层实现原理 CPU 缓存一致性 文章目录 【操作系统】CPU 缓存一致性、MESI 协议CPU Cache 的数据写入写直达写回 缓存一致性问题总线嗅探MESI 协议总结 CPU Cache …

springboot:缓存不止redis,学会使用本地缓存ehcache

0. 引言 随着redis的普及&#xff0c;更多的同学对redis分布式缓存更加熟悉&#xff0c;但在一些实际场景中&#xff0c;其实并不需要用到redis&#xff0c;使用更加简单的本地缓存即可实现我们的缓存需求。 今天&#xff0c;我们一起来看看本地缓存组件ehcache 1. ehcache简…

python+vue 市政工程资源互助平台的设计与实现

该系统将由用户系统&#xff0c;管理员系统两部分组成。用户有个人和vip两种类型。 用户模块包括登录注册功能&#xff0c;登录字段包括用户名&#xff0c;密码&#xff0c;用户身份以及验证码。注册包括用户名&#xff0c;密码&#xff0c;邮箱&#xff0c;电话号码&#xff0…

vite+vue+element-plus完成一个admin管理后台

整体项目的 访问链接&#xff1a;https://bigmiss.top/demo/index.html 用到的技术整理 名称版本安装命令说明vite4.0.0npm init vitelatest构建Vue项目vue3.2.45npm install vuenext渐进式框架&#xff08;在vite已安装&#xff09;element-plus3.7.5npm install element-plu…

C#调用C++封装的SDK库(dll动态库)——下

C#调用C封装的SDK库(dll动态库)——下 一、说明 上一篇我们相当于封装的是C语言风格的动态dll库&#xff0c;供C#来调用的。 C#调用C封装的SDK库(dll动态库)——上 如果我们要封装的是下面的类呢&#xff1f;我们该怎么办&#xff1f;大家先思考下。 class Calculation { p…

Tomcat常用操作

Tomcat时间长不用&#xff0c;居然不会用了&#xff0c;这里用的Tomcat9.0.74&#xff0c;对应的jdk版本是jdk8与jdk11。 先看他的把Tomcat安装上去&#xff1a;Tomcat9的安装 运行与关闭Tomcat winr输入cmd。在运行窗口输入startup.bat&#xff0c;回车&#xff0c;启动Tom…

Java基础:对象的克隆(复制)

假如想复制一个简单变量。很简单&#xff1a; int apples 5; int pears apples; 不仅int类型&#xff0c;其它七种原始数据类型(boolean,char,byte,short,float,double.long)同样适用于该类情况。 但是如果你复制的是一个对象&#xff0c;情况就复杂了。 假设说我是一个b…

webpack----开发服务器

文章目录 devServer抽取csscss的兼容性压缩cssjs语法检查js的兼容性 devServer 每次编辑源码后&#xff0c;都要webpack重新打包&#xff0c;才能看到效果&#xff0c;麻烦&#xff01;使用webpack-dev-server 自动打包编译源码配置 // webpack.config.js ... mode: "de…

shell中的for循环和if判断

一.编写脚本for1.sh,使用for循环创建20账户&#xff0c;账户名前缀由用户从键盘输入&#xff0c;账户初始密码由用户输入&#xff0c;例如: test1、test2、test3、.....、 test10 1.创建脚本for1.sh [rootserver ~]# vim for1.sh 2.编写脚本for1.sh 3.执行脚本for1.sh [roo…

linux命令----- mkdir与rmdir

创建与删除目录 一 mkdir1.mkdir 目录名2.mkdir -p 目录一/目录二 二 rmdir1.rmdir 目录名2.删除非空目录时失败3. rmkdir -p 目录1/目录2 一 mkdir mkdir是make directories的缩写&#xff0c;主要用于linux中创建目录 创建的目录不能和同级目录中已经存在的目录重名可以mkd…