【C语言】字符串处理函数:strtok和strerror

news2025/5/15 19:54:24

在C语言中,字符串处理是编程的基础之一。本文将详细讲解两个重要的字符串处理函数:strtokstrerror

一、strtok函数

strtok函数用于将字符串分割成多个子串,这些子串由指定的分隔符分隔。其原型定义如下:

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

 

  • 功能strtok会将字符串s按照delim中定义的分隔符进行分割,并返回指向第一个分隔符后字符的指针。
  • 第一个参数str指向一个字符串,它包括了0个或者多个由sep字符串中一个或多个分隔符分割的标记。
  • strtok函数找到str中的下一个标记,并将其用 \0结尾,返回一个指向这个标记的指针。(注:strtok函数会改变被操作的字符串,所以被strtok函数切分的字符串一般都是临时拷贝的内容并且可修改。)
  • strtok函数的第一个参数不为 NULL,函数将找到str中第一个标记,strtok函数将保存它在字符串中的位置
  • strtok函数的第一个参数为 NULL,函数将在同一个字符串中被保存的位置开始,查找下一个标记。
  • 如果字符串中不存在更多的标记,则返回 NULL 指针。
  • 第二个参数delim: 分隔符集合。可以是一个或多个字符,函数会根据这些字符分割字符串。
工作原理
  • strtok 会修改原始字符串:它将找到的分隔符替换为 \0(空字符),从而将原始字符串分割成多个子串。
  • 第一次调用时,函数会从 str 的开头开始查找分隔符;后续调用时,如果传入 NULL,函数会从上一次停止的位置继续查找。

 使用示例

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <string.h>
int main()
{
	char arr[] = "abcdefg@hhh.aaaa";
	char sep[30] = { 0 };
	strcpy(sep, arr);
	const char* tmp = "@.";
	char* p = strtok(sep, tmp);
	printf("%s", p);

	return 0;;
}

 这是已经保存了@的地址,如果想继续往下运行就用到了(strtok函数的第一个参数为 NULL,函数将在同一个字符串中被保存的位置开始,查找下一个标记。)

所以要把strtok中的第一个参数修改一下,改为NULL。

如下:

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <string.h>
int main()
{
	char arr[] = "abcdefg@hhh.aaaa";
	char sep[30] = { 0 };
	strcpy(sep, arr);
	const char* tmp = "@.";
	char* p = strtok(sep, tmp);//(abcdefg\0hhh.aaaa)
	printf("%s\n", p);
    p = strtok(NULL, tmp);//(abcdefg\0hhh\0aaaa)
	printf("%s\n", p);
	p = strtok(NULL, tmp);//(abcdefg\0hhh\0aaaa)
	printf("%s\n", p);

	return 0;;
}

 

注意事项

  1. 修改原始字符串

    • strtok 会直接修改原始字符串,因此如果需要保留原始数据,建议先复制一份副本再操作。
      char arr[] = "abcdefg@hhh.aaaa";
      	char sep[30] = { 0 };
      	strcpy(sep, arr);

      这里将arr的数据复制在sep中

  2. 线程安全问题

    • strtok 使用静态缓冲区存储当前分割位置,因此它不是线程安全的。在多线程环境中,建议使用线程安全的替代函数 strtok_r
  3. 连续分隔符处理

    • 如果字符串中有连续的分隔符(如 ,,,),strtok 会忽略这些连续的分隔符,不会返回空字符串。
    • "a,,b" → "a", "b" // 空字段被跳过

  4. 字符串常量不可修改

    char *str = "read-only"; // 错误!应使用char[]
  5. 跨调用干扰

    ​
    char *t1 = strtok(str1, ","); 
    char *t2 = strtok(str2, ";"); // 破坏str1的处理状态
    
    ​
  6. 空指针检查

    while ((token = strtok(NULL, ",")) != NULL)
    简化方案

 用上for循环,来省略繁琐步骤。

如下:

int main()
{
	char arr[] = "abcdefg@hhh.xxx";
	char sep[30] = { 0 };
    strcpy(sep, arr);
	const char* tmp = "@.";
	char* p = NULL;
	for (p = strtok(sep, tmp); p != NULL; p = strtok(NULL, tmp))
	{
		printf("%s\n", p);
	}
	return 0;
}

 

分析如下:

首次调用 strtok 函数,传入 sep 和 tmp 作为参数。strtok 会找到第一个分隔符 @,并将 sep 中该位置的字符替换为 \0,然后返回指向第一个子字符串 "abcdefg" 的指针。

这个for循环代码的判断条件是p不等于\0(而言就是NULL),当等于\0时,循环终止。

后续调用 strtok 函数时,传入 NULL 和 tmp 作为参数strtok 会继续从上次分割的位置开始查找下一个分隔符 .,并将该位置的字符替换为 \0,然后返回指向下一个子字符串 "hhh" 的指针。如果找不到更多的分隔符,则返回 NULL,循环结束。

二·stderror 函数

它的主要功能是将整数形式的系统级库级的错误码转换为可读性更强的描述性文本 。这种机制使得程序员能够更直观地理解并处理运行过程中发生的异常情况 。

工作原理
char *stderror (int errnum) 

注意:strerror生成的错误字符串取决于开发平台和编译器 

  • 参数

    • int errnum: 待转换的错误码 。通常情况下取值为全局变量 errno ,但理论上可以接受任意整型值 。
  • 返回值:该函数返回一个指向错误字符串的指针,该错误字符串描述了错误 errnum。

以下时返回值的意思。 

 

使用示例 
#include <stdio.h> 
#include <stdlib.h> 
#include <string.h> 
#include <errno.h> 

int main()
{

	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		printf("%s\n", strerror(errno));
		return 1;
	}
	return 0;
}

这里运用到打开文件,   (fopen是打开文件的函数,在后续会提到,此处不过多解释,前参数是文件名,后参数是打开的方式)(r是读的意思
 打印文件失败的时候,会返回NULL指针
 如果打开成功,返回的是非NULL指针,返回的pf中。
 以r(读)的形式打开文件,文件如果不存在,则打开失败。

return1;  返回非零值,表示程序执行过程中发生了错误

return 0;  如果文件打开成功,返回0,表示程序成功执行

 

显然在我的电脑中是没有这个文件的,无法打开。

注意事项 

1. 缓冲区覆盖问题

// 危险代码示例
char *err1 = strerror(EPERM);
char *err2 = strerror(ENOENT);
printf("%s, %s\n", err1, err2); // 可能输出相同内容

2. 无效错误码处理

printf("%s\n", strerror(9999)); // 可能输出:Unknown error 9999

3. 国际化问题

// 使用本地化版本
#include <libintl.h>
printf("%s\n", dgettext("glibc", strerror(errno)));

三·perror 函数 

perror 是 C语言标准库中的一个简单而实用的工具 。它的主要功能是根据当前全局变量 errno的值生成一条描述性错误消息 ,并将其输出到标准错误流 (stderr) 。与 stderror不同 , perror直接处理了消息的格式化与输出过程 ,使得程序员能够更便捷地报告错误 。(相当于printf+sderror)

工作原理
  1. 检查全局变量 errno的值 。
  2. 根据该值查找对应的错误描述文本 。
  3. 将用户提供的前缀(如果有)与错误描述拼接成完整消息 。
  4. 将最终结果写入标准错误流 (stderr) 。
void perror (const char *s) 
  • 参数

    • const char *s: 用户自定义的前缀字符串 。如果非空 ,该前缀会与系统提供的错误描述拼接在一起 ;如果为空 ,则仅打印系统提供的错误描述 。
  • 返回值

    • 无返回值 。
 使用示例
#include <stdio.h> 
#include <stdlib.h> 
#include <string.h> 
#include <errno.h> 

int main()
{

	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		printf("%s\n", strerror(errno));
		return 1;
	}
	return 0;
}

 

结语:

字符串处理函数:strtok和strerror讲解完毕,希望本章对大家有所帮助。

C语言的娴熟运用需要多练勤练,大家快去练习把!!

 

 

 

 

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

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

相关文章

go语言:开发一个最简单的用户登录界面

1.用deepseek生成前端页面&#xff1a; 1.提问&#xff1a;请你用html帮我设计一个用户登录页面&#xff0c;要求特效采用科技感的背景渲染加粒子流动&#xff0c;用css、div、span标签&#xff0c;并给出最终合并后的代码。 生成的完整代码如下&#xff1a; <!DOCTYPE h…

《Maven高级应用:继承聚合设计与私服Nexus实战指南》

一、 Maven的继承和聚合 1.什么是继承 Maven 的依赖传递机制可以一定程度上简化 POM 的配置&#xff0c;但这仅限于存在依赖关系的项目或模块中。当一个项目的多个模块都依赖于相同 jar 包的相同版本&#xff0c;且这些模块之间不存在依赖关系&#xff0c;这就导致同一个依赖…

C语言数字分隔题目

一、题目引入 编写一个程序,打印出从用户输入的数字开始,递减到1的序列。要求每次打印一行,数字之间用逗号分隔,最后一个数字后面没有逗号。 二、代码展示 三、运行结果 四、思路分析 1.先用一个for循环对输入的数字进行递减 2.再对for循环里面的数字进行筛选 如果大于1 …

ARM-外部中断,ADC模数转换器

根据您提供的图片&#xff0c;我们可以看到一个S3C2440微控制器的中断处理流程图。这个流程图展示了从中断请求源到CPU的整个中断处理过程。以下是流程图中各个部分与您提供的寄存器之间的关系&#xff1a; 请求源&#xff08;带sub寄存器&#xff09;&#xff1a; 这些是具体的…

git克隆数据失败

场景&#xff1a;当新到一家公司&#xff0c;然后接手了上一个同时的电脑&#xff0c;使用git克隆代码一直提示无法访问&#xff0c;如图 原因&#xff1a;即使配置的新的用户信息。但是window记录了上一个同事的登录信息&#xff0c;上一个同事已经被剔除权限&#xff0c;再拉…

自动化备份全网服务器数据平台

自动化备份全网服务器数据平台 项目背景知识 总体需求 某企业里有一台Web服务器&#xff0c;里面的数据很重要&#xff0c;但是如果硬盘坏了数据就会丢失&#xff0c;现在领导要求把数据做备份&#xff0c;这样Web服务器数据丢失在可以进行恢复。要求如下&#xff1a;1.每天0…

AI Agent系列(八) -基于ReAct架构的前端开发助手(DeepSeek)

AI Agent系列【八】 项目目标一、核心功能设计二、技术栈选择三、Python实现3.1 设置基础环境3.2 定义AI前端生成的类3.4 实例化3.5 Flask路由3.6 主程序执行 四、 功能测试 项目目标 开发一个能够协助HTMLJSCSS前端设计的AI Agent&#xff0c;通过在网页中输入相应的问题&am…

一文学会云服务器配置Milvus向量数据库

服务器准备 首先&#xff0c;我们需要进行服务器的准备&#xff0c;这里准备的是RTX-4090服务器 连接我们已经创建好的服务器&#xff0c;这里可使用MobaXterm进行ssh连接 ssh funhpcIP地址 一键完成Docker配置 注&#xff1a;docker的旧版本不一定被称为docker&#xff0c;doc…

19685 握手问题

19685 握手问题 ⭐️难度&#xff1a;简单 &#x1f31f;考点&#xff1a;2024、省赛、数学 &#x1f4d6; &#x1f4da; package test ;import java.util.Scanner; public class Main {public static void main(String[] args) {Scanner scanner new Scanner(System.in);…

【MySQL数据库】锁机制

概述 锁&#xff1a;是计算机协调多个进程或者线程并发访问某一资源的机制。在数据库中&#xff0c;除了传统的计算资源&#xff08;CPU、RAM、IO&#xff09;的争用以外。数据也是一种供多用户共享的资源。如何保证数据的并发访问的一致性、有效性是所有数据库必须解决的一个…

react redux的学习,单个reducer

redux系列文章目录 一 什么redux&#xff1f; redux是一个专门用于做状态管理的JS库(不是react插件库)。它可以用在react, angular, vue等项目中, 但基本与react配合使用。集中式管理react应用中多个组件共享的状 简单来说&#xff0c;就是存储页面的状态值的一个库&#xf…

分布式控制技术赋能智慧工厂精准控制研究

摘要&#xff1a;本文聚焦于分布式控制技术在智慧工厂精准控制中的应用。详细阐述了分布式控制系统&#xff08;DCS&#xff09;、边缘计算机、边边协同技术以及分布式计算等关键要素在实现精准控制中的作用机制。同时&#xff0c;分析了云边协同模式存在占用带宽、单点故障、数…

清明节里清明菜:软萩(拟人版介绍)

好像人们无论过任何节&#xff0c;总是离不开吃 清明节里吃清明菜&#xff0c;你采摘了吗&#xff1f; 姓名 软萩&#xff0c;也叫鼠麴草、清明菜、软雀&#xff0c;学名鼠曲草。 一些地方性小名&#xff08;防止大家找不到组织&#xff0c;已知的都附上&#xff09;&#xf…

JavaWeb学习--MyBatis-Plus整合SpringBoot的ServiceImpl方法(增加,修改与删除部分)

接下来是常用的增加&#xff0c;修改以及删除部分 首先是增加部分&#xff0c;增加一个新的数据 Testpublic void testInsert() {// 添加一个新用户记录Student s new Student();s.setName("NewStudent");s.setAge(25);boolean saved studentService.save(s);//可以…

AiCube 试用 - 创建流水灯工程

AiCube 试用 - 创建流水灯工程 本文介绍了 Aiapp-ISP 仿真调试平台软件的 AiCube 工具&#xff0c;实现流水灯工程的快速创建的主要流程。 下载运行 下载 最新版 AIapp-ISP 软件&#xff1b; 解压并打开该软件&#xff0c;右侧操作界面选择并进入 Keil 仿真设置 标签项&…

GISBox:核心功能免费的一站式三维GIS处理平台

大家好&#xff0c;今天为大家介绍的软件是GISBox&#xff1a;一款核心功能免费的一站式三维GIS处理平台&#xff0c;主要是适用于数字孪生。下面&#xff0c;我们将从软件的主要功能、支持的系统、软件官网等方面对其进行简单的介绍。 软件官网&#xff1a;http://www.gisbox.…

【论文笔记】DeepSeek-R1 技术报告

最强开源LLM&#xff0c;性能和效果都很棒&#xff1b;在数学、代码这种有标准正确答案的场景&#xff0c;表现尤为突出&#xff1b;一些其他场景的效果&#xff0c;可能不如DeepSeek-V3和Qwen。 Deepseek-R1没有使用传统的有监督微调sft方法来优化模型&#xff0c;而使用了大规…

基于javaweb的SSM羽毛球会员俱乐部系统场馆课程运动设计与实现(源码+文档+部署讲解)

技术范围&#xff1a;SpringBoot、Vue、SSM、HLMT、Jsp、PHP、Nodejs、Python、爬虫、数据可视化、小程序、安卓app、大数据、物联网、机器学习等设计与开发。 主要内容&#xff1a;免费功能设计、开题报告、任务书、中期检查PPT、系统功能实现、代码编写、论文编写和辅导、论文…

[dp4_路径问题] 下降路径最小和 | 最小路径和 | 地下城游戏

目录 1.下降路径最小和 题解 2.最小路径和 题解 3.地下城游戏 题解 做算法题的时候&#xff0c;谨记图画得越详细越好&#xff0c;思路想的越清晰越好&#xff0c;然后再用代码实现一下就好啦 1.下降路径最小和 链接&#xff1a;931. 下降路径最小和 给你一个 n x n 的…

EasyExcel 数据字典转换器实战:注解驱动设计

一、场景痛点与解决方案 1. 问题背景 在 Excel 导入导出场景中&#xff0c;开发者常面临以下问题&#xff1a; 数据可读性差&#xff1a;数据库存储的字典值&#xff08;如 1、true&#xff09;直接导出时难以理解双向转换复杂&#xff1a;导入时需将用户输入的标签反向解析…