C语言函数: 字符串函数及模拟实现strtok()、strstr()、strerror()

article/2025/10/14 17:00:08

C语言函数: 字符串函数及模拟实现strtok()、strstr()、strerror()

strstr()函数:

        作用:字符串查找。在一串字符串中,查找另一串字符串是否存在。

形参:

4b2e6b20082d41528eadfab74266debc.png

       str2在str1中寻找。返回值是char*的指针

        原理:如果在str1中找到了str2,则返回在str1中存在的str2的字符串的第一个字符的地址。如果在str1中每找到str2,则返回NULL(空指针)。

举例:

#define _CRT_SECURE_NO_WARNINGS#include <stdio.h>int main()
{char arr1[] = "abcdefgh";char arr2[] = "bcde";char arr3[] = "bcdeh";char* a1 = strstr(arr1, arr2);char* a2 = strstr(arr1, arr3);return 0;
}

        a1是在arr1中寻找arr2,在arr1[1]~arr1[4]上找到了与arr2一样的字符串,则返回arr1中字符'b'的地址。

        a2是在arr1中寻找arr3,在arr1中并未找到与arr1一样的字符串,则返回NULL。

#define _CRT_SECURE_NO_WARNINGS#include <stdio.h>int main()
{char arr1[] = "abcdefgh";char arr2[] = "bcde";char arr3[] = "bcdeh";char* a1 = strstr(arr1, arr2);char* a2 = strstr(arr1, arr3);if (a1 != NULL){printf("arr2找到了");//结果:arr2找到了}if (a2 == NULL){printf("arr3没找到");//结果:arr3没找到}return 0;
}

strstr()函数的模拟实现:

char* my_strstr(const char* str1, const char* str2)//因为两个地址值不会改变,所以加const
{assert(str1 && str2);//防止两个参数为NULLchar* pt1 = NULL;char* pt2 = NULL;char* nc = str1;//用于在字符串中找到与str2首元素一样字符。while (*nc)//当str1中找完了,都没有找到和str2中首元素一样的字符时停止循环。{pt1 = nc;//用于定位到每次找到和str2中首元素一样的字符的地址。pt2 = str2;//用于指向str2的首元素地址。while (*pt1 && *pt2 && ( * pt1 == *pt2))//找到和str2中首元素一样的字符{pt1++;pt2++;}if (*pt2 == '\0')//当上面的循环走完,ptr2的指向\0时,说明在str1中找到了和str2一样的字符串{return nc;//返回当时的字符串。}nc++;//当这个str1中的元素和str2不匹配,就要看下一个字符是否匹配。}return NULL;
}

        

int main()
{char arr1[] = "abbcdef";char arr2[] = "bcd";char* a1 = my_strstr(arr1, arr2);if (a1 != NULL){printf("arr2找到了\n");//结果:arr2找到了}else{printf("arr2没找到\n");}return 0;
}

        

补充:还有一种算法:KMP算法。也是一个字符串查找的算法。


strtok()函数:

        作用:对字符串进行切分(分割)

        #include <string.h>

参数:

a8252c78062a410c91484cd68d05255b.png

        str是将被分割的字符串

        sep是需要传入的是字符串,sep内包含了需要作为分割符的字符的集合

        71ff5a9b43c14eddb52b3d06b567d768.png

        如果要将str中分为三个部分:www、qq、com

        那么就需要在@和.处进行分割。因此,@和.就是分割符,将分隔符放入sep字符串中,此时就是 :分割符的字符的集合

       

原理:

       这里有个字符串str :"www@qq.com\0"

        strtok()会将按照sep所指示的,在str中寻找sep内存在的字符。

        找到@后,会将@改成\0。

        strtok()返回@之前的字符串的首元素地址,即:str中第一个'w'的地址。

        并且strtok会记录此时\0的位置。

        strtok有两种机制:

        1.如果strtok不为NULL,那么会保存@的位置(@会被改成\0)     

        2.如果strtok为 NULL,那么会从上次strtok所保存@的位置,继续往下找seq中的字符,如果找到,接着分割。最后再保存被改成\0的位置。

        最后:如果strtok找到了没被改的\0,也就是字符串末尾的\0时,返回NULL。

        

        那么有疑问了,strtok是怎么保存@的位置呢?

        其实,想要保存某个数据,实际上就是创建一个全局变量。只要创建了全局变量,程序运行到哪里,无论在哪个函数中,都可以被使用。

        因此,strtok就是利用全局变量的能力,从而保存了@的位置。那么strtok是怎么保存的呢?难道直接在代码中写上一个全局变量吗?显然不是。

        strtok其实是使用static修饰了存放@位置的变量,因为static基本含义就是让一个变量成为全局变量。

        static函数名除了对该函数声明的文件可见外,其他文件都无法访问。也就是只能被本文件中的函数调用,而不能被同一程序中的其他文件的函数调用。

        static可以限定变量或函数为静态存储。静态函数会被自动分配在一个一直使用的存储区,直到程序结束才从内存消失,static限定的变量或函数不会和同一程序中其他文件同名的相冲突。如果用static限定内部变量,则该变量从程序一开始就用有内存,不会随其所在函数的调用和退出而分配和消失。

        

#define _CRT_SECURE_NO_WARNINGS#include <stdio.h>
#include <assert.h>void cc()
{static int opp = 0;opp++;printf("%d", opp);
}int main()
{cc();//结果:1int opp = 3;cc();//结果:2return 0;
}

使用建议:

        因为strtok的会对str字符串本身进行修改,如果不想对strtok进行修改时,可以用strcpy进行拷贝,然后对拷贝的字符串使用strtok

        

char arr[] = "www@qq.com";char* p = "@.";char tmp[20] = { 0 };strcpy(tmp, arr);//拷贝

实验:

        

#define _CRT_SECURE_NO_WARNINGS#include <stdio.h>
#include <assert.h>
#include <string.h>int main()
{char arr[] = "www@qq.com";char* p = "@.";char tmp[20] = { 0 };strcpy(tmp, arr);char* ret = NULL;ret = strtok(tmp, p);printf("%s\n", ret);//wwwret = strtok(NULL,p);printf("%s\n", ret);//qqret = strtok(NULL, p);printf("%s\n", ret);//comret = strtok(NULL, p);printf("%s\n", ret);//(null)return 0;
}

        创建了一个ret用于存放,每次strtok返回的字符串首元素地址。

        第一次调用 strtok(tmp, p)拿到了(返回)'w'的地址,保存了@(\0)的位置。

        第二次调用strtok(NULL,p)拿到了(返回)'q'的地址,保存了.(\0)的位置。

        第三次调用strtok(NULL,p)拿到了(返回)'c'的地址,\0的位置。

        第四次调用strtok(NULL,p)拿到了(返回)NULL,这是因为\0后面找不到p内包含的字符了。

        结果打印三个部分:www、qq、com

代码强化:

        这一个一个地调用strtok显然是不合理地。

        并且我们发现除了第一次调用strtok,传入地第一个参数是一个活指针之外,后面的几次调用使用的都是NULL,因此我们可以创建一个循环,从而分割它们。

        

#define _CRT_SECURE_NO_WARNINGS#include <stdio.h>
#include <assert.h>
#include <string.h>int main()
{char arr[] = "www@qq.com";char* p = "@.";char tmp[20] = { 0 };strcpy(tmp, arr);char* ret = NULL;for ( ret = strtok(tmp, p); ret != NULL; ret = strtok(NULL, p)){printf("%s\n", ret);}return 0;
}

        for(初始化值;判断条件;对值改变)

        初始化值,执行了strtok第一次调用。

        判断条件,如果为strtok返回了NULL,说明没有分隔符了,for循环就停止。

        对值改变,做到了除第一次以外,后n次地对strtok地调用。

        

模拟实现:

       C语言源码剖析与实现——strtok()系列函数实现_strtok源码_C+G的博客-CSDN博客


strerror

        作用:翻译错误码。在使用库函数的时候,如果失败了都会出现错误码,而这串错误码你完全不懂是什么意思,比如:5,这个错误码的意思是什么呢?肯定不知道吧。

        错误码的作用是告诉你,你的代码执行过程中出现的错误。比如内存空间不足等等...

参数:

        头文件:#include <string.h>

d954914288674d1ebd423beeb356dcfd.png

        strerror返回的是一个地址,这个地址指向了一个字符串。这个字符串就是strerror从错误码所翻译出来的一句话。

        

#define _CRT_SECURE_NO_WARNINGS#include <stdio.h>
#include <assert.h>
#include <string.h>
#include <errno.h>int main()
{printf("%s\n", strerror(0));printf("%s\n", strerror(1));printf("%s\n", strerror(2));return 0;
}

 

        除此之外,还有许多C语言内部的错误码。

errno:

        errno是C语言定义的一个全局变量,它的作用是存放代码执行过程中出现的错误码。

        它不需要人为的定义,当用到它的时候只需要引头文件:#include <errno.h>

        

举例:

        

#define _CRT_SECURE_NO_WARNINGS#include <stdio.h>
#include <assert.h>
#include <string.h>
#include <errno.h>int main()
{//fopen:打开文件FILE* pf = fopen("test.txt", "r");//test.txt不存在if(pf ==NULL)//fopen找不到test.txt会返回空指针printf("%s\n", strerror(errno));//返回错误信息return 0;
}

模拟实现:

        单纯的翻译代码,因此不做实现。

 


http://chatgpt.dhexx.cn/article/yMDWATON.shtml

相关文章

【strerror函数的使用】

strerror函数的使用 一&#xff0c;说明二&#xff0c;具体使用场景 一&#xff0c;说明 strerror会返回错误码&#xff0c;我们可以将其翻译成所对应的错误信息&#xff1b; c语言的库函数在运行的时候&#xff0c;如果发生错误&#xff0c;就会将错误码存放在一个变量中&…

strerror 函数

收藏 75 23 strerror编辑 本词条缺少 名片图&#xff0c;补充相关内容使词条更完整&#xff0c;还能快速升级&#xff0c;赶紧来 编辑吧&#xff01; 通过标准错误的标号&#xff0c;获得错误的描述字符串 &#xff0c;将单纯的错误标号转为字符串描述&#xff0c;方便用户查找…

strerror函数介绍

认识strerror 库函数调用失败的时候会产生错误码&#xff0c;而每一个错误码对应着一条错误信息&#xff0c;strerror函数的作用就是将错误码给转化成错误信息。 在C语言中有一条全局的错误码errno&#xff0c;在程序运行过程中&#xff0c;只要库函数调用失败&#xff0c;我们…

strerror perror

strerror这个函数把错误码转化为错误信息,把错误信息的起始地址返回 X86下的代码 #define _CRT_SECURE_NO_WARNINGS 1 #include<stdio.h> #include<string.h> int main() { printf("%s\n", strerror(0)); printf("%s\n", strerror(1…

关于strerror

1功能 strerror用于返回错误码信息的首字符的地址。 可以这样理解&#xff0c;函数运行过程中如果失败的话&#xff0c;会返回一个错误码放在errno中&#xff0c;&#xff08;原先设计中就已经存在的&#xff09;调用strerror函数的话&#xff0c;可以打印对应的错误信息 2 …

Multipass虚拟机入门教程

目录 一、Multipass介绍 二、Multipass的安装 1. 系统版本 2. 下载地址和官网教程 3. 安装 4. 查看版本并测试是否安装成功 5. 若启动失败 三、Multipass的简单使用 1. 设置虚拟化提供方式&#xff08;忽略该步骤&#xff09; 2. 创建名为test虚拟机&#xff0c;分配…

服务器vmware新建虚拟机教程,如何创建虚拟机教程全解

这部分教程我们将学习的是如何创建虚拟机。在创建虚拟机之前&#xff0c;vSphere Client是必要的软件之一&#xff0c;它用于访问ESX主机或vCenter的图形管理用户界面。 vSphere Client安装在Windows计算机上&#xff0c;它是与虚拟基础架构进行交互的主要方法。 1.安装vSphere…

专为折腾而生!老旧电脑安装PVE虚拟机保姆教程

专为折腾而生&#xff01;老旧电脑安装PVE虚拟机保姆教程 这几天玩VMware虚拟机上瘾&#xff0c;感觉特别有意思。然而我其实并不满足于只是在这种软件层面上玩玩&#xff0c;而想挑战更高级的玩法&#xff0c;比如说玩玩可以安装在实体机上的虚拟机系统~~ 说直接点就是我想在…

虚拟机Hyper-V的安装以及使用教程

目录 前言 一、什么是虚拟机Hyper-V&#xff1f; 二、使用步骤 虚拟机的安装 前言 Hyper-V是Windows操作系统提供的虚拟机管理平台&#xff0c;所有提供Hyper-V功能的Windows操作系统都可以根据本论文的内容安装并管理虚拟机。具体来说&#xff0c;Hyper-V 提供硬件虚拟化…

VirtualBox基础使用教程

选择VirtualBox而不是VMware Workstation的原因:VirtualBox是开源软件,对于个人的Linux学习来说,既免费又够用,且相较于VMware Workstation来说更为小巧,也没有那么多的自启动服务,不会在你不使用虚拟机的时候也有那么多服务在后台运行。 下载并安装VirtualBox 注:以下…

VMWare安装Linux虚拟机详细教程

使用VMware创建Linux并配置网络 1. 创建虚拟机&#xff0c;选择自定义 2. 选择硬件兼容性 3. 选择稍后安装操作系统 4. 选择Linux系统 5. 创建虚拟机名称和位置 6. 根据需求选择处理器 8. 选择虚虚拟机内存 9. 选择虚拟机的网络类型 10. 选择I/O控制器类型 11. 选择磁盘类型 1…

华为虚拟机服务器怎么使用教程,虚拟机装服务器教程

虚拟机装服务器教程 内容精选 换一换 应用容器化改造有三种方式,您可单击这里查看。本教程以某游戏为例,将该游戏进行微服务的架构改造,再进行容器化。本教程不对改造细节做深度讲解,仅讲解大致的建议。如需要详细了解容器化改造的过程,请单击服务咨询。本章节主要讲解游戏…

解决oracle数据库登录缓慢的问题

在连接tmboot开启服务的时候&#xff0c;tmboot初始化进程十分缓慢&#xff0c;于是查看日志和源码&#xff0c;发现是启动进程的时候&#xff0c;需要连接数据库进行初始化。 但是我本身在连接数据库的时候&#xff0c;用conn 连接对应数据库的时候都需要十秒才能连接上&…

数据库的登陆方法

数据库的两种登陆方法 作者&#xff1a;吴炳耀 完成时间&#xff1a;2019.1.18 当我们进入数据库系统时&#xff0c;服务器类型一般都是选数据库引擎&#xff0c;因为其他类型我们用到的机会是非常少的 服务器名称在我们安装时就已自动分配好,如果没有我们可以输入&#xff1a;…

【数据库】浅析登录名与用户的联系

前言&#xff1a;当我们学到“数据库安全性控制”这章时&#xff0c;会遇到登录名与用户这两个概念。如果没有理清楚&#xff0c;那么在SQL Server 软件中会不知如何下手。本篇文章就是来浅析他们之间的关系以及再该软件中如何创建。 一、登录名 我们登陆SQL Server软件时&am…

用命令行登录并操作数据库

一、数据库的登录及数据定义 1、登录到数据库 首先打开Mysql安装目录中的bin目录&#xff0c;之后在地址栏输入cmd&#xff0c;回车。我的是安装在了C盘。 ​​​​ 进入到了命令行界面&#xff0c;并输入mysql -u root -p&#xff0c;回车后再输入密码&#xff0c;显示如下界…

C#连接数据库的登录界面

首先建立一个数据库&#xff0c;是在Thisone的数据库里建立的一个Resign表&#xff0c;里面有两个变量&#xff0c;一个是SID一个是Skey&#xff0c;就是用户名和密码。 下面打开visual studio&#xff0c;新建项目&#xff0c;搭建好自己的界面。 我这里就是简单的登录界面&am…

达梦数据库新创建用户登录报错“登录失败次数超过限制”

问题描述 最近遇到数据库新创建A用户&#xff0c;登录失败的情况。创建后使用A用户第一次登录报错&#xff1a;”登录失败次数超过限制“的问题。多次登录用户被锁定后&#xff0c;解锁后登录还是相同的错提示。数据库版本&#xff1a;DM8 1-1-156 排查步骤 1、新建实例&…

Android 连接数据库实现登录注册功能(SQLite)

什么是SQLite SQLite 是一个软件库&#xff0c;实现了自给自足的、无服务器的、零配置的、事务性的 SQL 数据库引擎。SQLite 是在世界上最广泛部署的 SQL 数据库引擎。SQLite 源代码不受版权限制。 以下是一些关于SQLite学习的有用的网站&#xff1a; 1、SQLite Home Page - …

达梦数据库入门

测试版本&#xff1a;达梦6 主要内容&#xff1a;数据库登录、数据库创建、用户创建、授权等。 说明&#xff1a;数据库有两种方式常用的使用方式&#xff0c;有达梦自带的比较友好的达梦管理工具&#xff0c;比较适用于新手。这里主要介绍的是第二种方式,命令行方式。 一、达梦…