C语言---常见的字符函数和字符串函数介绍

目录

前言

1 字符分类函数

2 字符转换函数

3 strlen的使用和模拟实现 

3.1 strlen的模拟实现 

4 strcpy的使用和模拟实现

4.1  strcpy的模拟实现

5 strcat的使用和模拟实现 

 5.1 strcat的模拟实现

 6 strcmp的使用和模拟实现

6.1 strcmp的模拟实现

7 strncpy函数的使用 

8 strncat函数的使用 

 9 strncmp函数的使用

10 strstr的使用和模拟实现 

10.1 strstr的模拟实现

11 strtok函数的使用 

12 strerror函数的使用 

 总结


前言

在写C语言程序时,我们常常要处理字符和字符串,这就需要利用到字符函数和字符串函数,C语言标准库中提供了一系列库函数,下面就介绍一下常见的字符函数和字符串函数。

1 字符分类函数

C语言中有一些函数是做字符分类的,意思就是辨别一个字符属于什么字符。这些函数在使用时要包含头文件<ctype.h>

函数如果函数参数符合下列条件就返回真
iscntrl任何控制字符
isspace空白字符:空格,‘’,换页‘\f’,换行‘\n’,回车‘\r’,制表符‘\t’,垂直制表符‘\v’
isdigit十进制数字0~9
isxdigit十六进制数字,包括所有十进制数字,小写字母a~f,大写字母A~F
islower小写字母a~z
isupper大写字母A~Z
isalpha字母a~z或A~Z
isalnum字母或者数字,a~z,A~Z,0~9
ispunct标点符号,任何不属于数字或者字母的图形字符(可打印)
isgraph任何图形字符
isprint任何可打印字符,包括图形字符和空白字符

这些函数使用方法类似,在这里举一个例子,其他函数使用同理:

int islower(int c);

islower是能够判断参数部分的c是否为小写字母,通常通过返回值来表明:如果是小写字母就返回非0的整数;如果不是小写字母,则返回0。

举例:写一个代码,将字符串中的小写字母转大写,其他字符不变 。代码如下:

#include <stdio.h>
#include <ctype.h>
int main()
{char str[] = "hELlO WorlD";char* p = str;while(*p != '\0'){if (islower(*p)){*p -= 32;}p++;}printf("%s\n",str);return 0;
}

输出结果:

 

2 字符转换函数

C语言中提供了2个字符转换函数:

1 int tolower(int c);//将参数传进去的大写字母转小写
2 int toupper(int c);//将参数传进去的小写字母转大写

上面写的小写转大写代码,是用-32实现的,有了转换函数,就可以直接利用toupper实现,如下:


#include <stdio.h>
#include <ctype.h>
int main()
{char str[] = "hELlO WorlD";char* p = str;while (*p != '\0'){if (islower(*p)){*p=toupper(*p);}p++;}printf("%s\n", str);return 0;
}

3 strlen的使用和模拟实现 

在C语言中,strlen是用来求字符串长度的,语法如下:

size_t strlen(const char* str);
  • 字符串以‘\0’为结束标志,strlen函数返回的是在字符串‘\0’前面出现的函数(不包含‘\0’)。
  • 参数指向的字符必须以‘\0’结束。
  • 注意函数的返回值是size_t,是无符号的。
  • strlen的使用需要包含头文件<string.h>。

 下面看一段代码:

#include <stdio.h>
#include <string.h>
int main()
{const char* str1 = "abcdef";const char* str2 = "bbb";if (strlen(str2) - strlen(str1) > 0){printf("str2>str1\n");}else{printf("str1>str2\n");}return 0;
}

这个代码输出结果是多少呢?这时候我们推测一下:str2长度是3,str1长度是6,str2长度减去str1的长度小于0,那么结果是srt1>str2,答案是这样吗?程序运行看一下结果:

输出结果:

结果恰恰相反,这是因为strlen的返回值是size_t,是无符号的,返回值是一个正数 。

3.1 strlen的模拟实现 

方法一,计数法:

int my_strlen(const char* str)//计数法
{assert(str);int count = 0;while (*str != '\0'){count++;str++;}return count;
}

方法二,指针-指针: 

int my_strlen(const char* str)//指针-指针
{assert(str);char* p = str;while (*str != '\0'){str++;}return str - p;
}

方法三 ,递归法:

int my_strlen(const char* str)//递归法
{assert(str);if (*str == '\0')return 0;elsereturn 1+my_strlen(str+1);
}
int main()
{char str[] = "hello world";int ret=my_strlen(str);printf("%d\n", ret);return 0;
}

4 strcpy的使用和模拟实现

基本语法:

char* strcpy(char* destination,const char* source)
  • 将source指向的字符串复制到destination指向的数组中,包括终止字符(并在该点停止)。
  • 源字符必须以‘\0’结束。
  • 会将源字符串中的‘\0’拷贝到目标空间。
  • 目标空间必须足够大,以确保能存放字符串。
  • 目标空间必须可修改 。

 举例:

#include <stdio.h>
int main()
{char str1[30] = "xxxxxxxxxxxx";const char str2[] = "hello world";strcpy(str1, str2);printf("%s\n", str1);return 0;
}

输出结果:

4.1  strcpy的模拟实现

char* my_strcpy(char* dest, const char* src)
{assert(dest != NULL);assert(src != NULL);char* ret = dest;while (*dest++ = *src++){;}return ret;
}

5 strcat的使用和模拟实现 

基本语法:

char * strcat ( char * destination, const char * source );
  • 将源字符串的内容附加到目标字符串,目标字符串结束的空字符将被源字符串的第一个字符覆盖。
  • 源字符串必须以‘\0’结束。
  • 目标字符串也得有‘\0’,否则没办法知道追加从哪里开始。
  • 目标空间必须有足够大,能容纳下源字符串的内容。
  • 目标空间必须可以修改。

举例:

#include <stdio.h>
int main()
{char str1[30] = "xxxxxxxxxxxx";const char str2[] = "hello world";strcat(str1, str2);printf("%s\n", str1);return 0;
}

输出结果:

 5.1 strcat的模拟实现

char* my_strcat(char* dest, const char* src)
{assert(dest != NULL);assert(src != NULL);char* ret = dest;while (*dest){dest++;}while (*dest++ = *src++){;}return ret;
}

 6 strcmp的使用和模拟实现

基本语法:

int strcmp ( const char * str1, const char * str2 );
  • 函数从两个字符串的首字符开始逐对比较。如果字符相同,则继续比较后续字符,直到遇到字符差异或终止空字符为止。 
  • 第一个字符串大于第二个字符串,则返回大于0的数字。
  • 第一个字符串等于第二个字符串,则返回0。
  • 第一个字符串小于第二个字符串,则返回小于0的数字。
  • 判断两个字符串的大小,比较两个字符串中对应位置上字符ASCII码值大小。

6.1 strcmp的模拟实现

#include <stdio.h>
#include <assert.h>
int my_strcmp(const char* str1, const char* str2)
{assert(str1 && str2);while (*str1 == *str2){if (*str1 == '\0')return 0;str1++;str2++;}return *str1 - *str2;
}
int main()
{char str1[] = "hello world";char str2[] = "hello type";int ret = my_strcmp(str1, str2);printf("%d", ret);return 0;
}

7 strncpy函数的使用 

基本语法:

 char * strncpy ( char * destination, const char * source, size_t num );
  • 这个函数将源字符串的前 num 个字符复制到目标位置。如果在复制完 num 个字符之前遇到源字符串的结尾(以空字符为标志),则目标位置将用零填充,直到总共写入 num 个字符为止。 
  • 拷贝num个字符从源字符串到目标空间。
  • 如果源字符串的长度小于num,则拷贝完源字符串之后,在目标的后边追加0,直到num个。

8 strncat函数的使用 

 基本语法:

char * strncat ( char * destination, const char * source, size_t num );
  • 将源字符串指向字符串的前num个字符追加到目标字符串的结尾,再追加一个\0字符。
  • 如果源字符串指向的字符串的长度小于num时,只会将字符串中到\0的内容追加到目标字符串的末尾。

 9 strncmp函数的使用

基本语法:

int strncmp ( const char * str1, const char * str2, size_t num );

比较str1和str2的前num个字符,如果相等则继续往后比较,最多比较num个字母,如果提前发现不一样,就提前结束,大的字符所在的字符串大于另外一个。如果num个字符都相等,返回0。

10 strstr的使用和模拟实现 

 基本语法:

char * strstr ( const char * str1, const char * str2);
  • 函数返回字符串str2在字符串str1中第一次出现的位置。
  • 字符串的比较匹配不包含\0字符,以\0作为结束标志。
#include <stdio.h>
#include <string.h>
int main()
{char str1[] = "abbcbbcdefbbcdef";char str2[] = "bbcdefb";char* ret = strstr(str1, str2);printf("%s\n", ret);return 0;
}

输出结果:

10.1 strstr的模拟实现

#include <stdio.h>
#include <assert.h>
char* my_strstr(const char* str1, const char* str2)
{assert(str1 && str2);const char* p = str1;const char* s1 = NULL;const char* s2 = NULL;if (*str2 == '\0')//检查空字符串return (char*) str1;while (*p){s1 = p;s2 = str2;while (*s1!='\0' && *s2 != '\0' && *s1 == *s2){s1++;s2++;if (*s2 == '\0')return (char*) p;}p++;}return(NULL);
}
int main()
{char str1[] = "abbcbbcdefbbcdef";char str2[] = "bbcdefb";char* ret = my_strstr(str1, str2);printf("%s", ret);return 0;
}

11 strtok函数的使用 

 基本语法:

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

举例:

#include <stdio.h>
#include <string.h>
int main()
{char arr[] = "192.168.11.22";char* str = NULL;char* sep = ".";for (str = strtok(arr, sep); str != NULL; str = strtok(NULL, sep)){printf("%s\n", str);}return 0;
}

 输出结果:

12 strerror函数的使用 

 基本语法:

char * strerror ( int errnum );

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

在不同的系统和C语言标准库的实现中都规定了⼀些错误码,⼀般是放在 <errno.h> 这个头文件中说明的,C语言程序启动的时候就会使用⼀个全面的变量errno来记录程序的当前错误码,只不过程序启动的时候errno是0,表示没有错误,当我们在使用标准库中的函数的时候发生了某种错误,就会将对应的错误码存放在errno中,一个错误码的数字是整数很难理解是什么意思,所以每⼀个错误码都是有对应的错误信息的。strerror函数就可以将错误对应的错误信息字符串的地址返回。下面编写代码打印错误信息:

#include <stdio.h>
#include <string.h>
#include <errno.h>
int main()
{int i = 0;for (i = 0; i <= 10; i++){printf("%d  %s\n", i,strerror(i));}return 0;
}

输出结果:

 总结

本文介绍了常见的字符函数和字符串函数,希望对你有用,如果这篇文章对你有用,可以点点赞哦,你的支持就是我写下去的动力,后续会不断地分享知识。

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

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

相关文章

Minio入门+适配器模式(实战教程)

一、安装Minio 1.1 拉取镜像 docker pull minio/minio docker images 1.2创建挂载目录 1.2.1 创建数据目录 mkdir -p /docker-minio/data 1.2.2 创建配置文件目录 mkdir -p /docker-minio/config 1.2.3 设置权限 chmod -R 777 /docker-minio/data /docker-minio/config …

LLaMA-Factory 对 omnisql 进行 ppo dpo grpo nl2sql任务 实现难度 时间 全面对比

在LLaMA-Factory框架下&#xff0c;针对omnisql任务&#xff08;自然语言到SQL生成&#xff09;应用PPO、DPO、GRPO三种算法的实现难度、时间及全面对比如下&#xff1a; 一、实现难度对比 1. PPO&#xff08;近端策略优化&#xff09; 难度&#xff1a;★★☆☆☆&#xff…

Kingbase 数据库中的 sys_guid() 函数报错

解决 Kingbase 数据库中的 sys_guid() 函数报错问题 问题背景 Kingbase 数据库在迁移或使用过程中&#xff0c;可能会遇到 select sys_guid() 函数报错 , 提示函数不存在的情况&#xff0c;这通常是由于以下几种原因造成的&#xff1a; 函数未正确安装或未启用函数参数不符合…

零基础RT-thread第五节:电容按键(2)

上一章的电容按键完全使用的HAL库的代码&#xff0c;并没有使用线程。这里尝试使用线程来控制电容按键。 依旧是 F767 本来以为会很容易实现&#xff0c;没想到尝试了很久&#xff0c;电容按键一直没有反应。 static rt_uint32_t measure_charge_time(void) {// 步骤1: 放电 …

华为云Flexus+DeepSeek征文|单机部署 与 CCE 高可用部署下 Dify 性能实测

引言 在当今的 AI 应用开发领域&#xff0c;选择合适的部署方式对于应用的性能表现、资源利用和成本控制至关重要。华为云为开发者提供了多样化的部署选择&#xff0c;其中基于单机 Flexus 实例的基础版部署和基于 CCE 容器的高可用版部署是两种常见的方式。本文将深入对比这两…

钉钉小程序框架:Pinia 状态管理与持久化存储封装

上一篇文章完成了 Pinia 在钉钉小程序中的引入与基础配置 文章地址&#xff1a;钉钉小程序框架引入 Pinia 状态管理-CSDN博客 本文将深入探讨如何通过Pinia 结合持久化存储 实现用户状态 在上一章节中&#xff0c;我们已经完成了 Pinia 在钉钉小程序中的引入与基础配置。本章将…

云计算产业链

一、云计算定义与分类体系 本质特征 按需服务模式&#xff1a;以网络化方式提供可配置的计算资源共享池&#xff08;网络/服务器/存储/应用&#xff09;。核心能力&#xff1a;快速弹性扩容、资源池化共享、按使用量付费、低管理开销。技术原理&#xff1a;通过分布式计算将大型…

git使用详解和示例

什么是 Git&#xff1f; Git 是一个 分布式版本控制系统&#xff08;DVCS&#xff09;&#xff0c;用于跟踪文件的变化&#xff0c;协调多人协作开发。由 Linus Torvalds 开发&#xff0c;用于管理 Linux 内核代码。 Git 的核心概念 名称说明工作区 (Working Directory)你看到…

深度学习的引出

虽然我们的神经⽹络给出了令⼈印象深刻的表现&#xff0c;但这样的表现带有⼏分神秘 ⽹络中的权重和偏置是被⾃动发现的。这意味着我们不能⽴即解释⽹络怎么做的、做了什么。我们能否找 到⼀些⽅法来理解我们的⽹络通过什么原理分类⼿写数字&#xff1f;并且&#xff0c;在知道…

GEO(生成式引擎优化)—— 内容创作者与企业的生死新战场

在搜索引擎优化&#xff08;SEO&#xff09;定义了互联网信息获取规则数十年后&#xff0c;一场由生成式人工智能&#xff08;AIGC&#xff09;驱动的风暴正悄然重塑整个格局。当ChatGPT、Claude、Gemini等AI助手能够直接生成整合后的答案&#xff0c;而非仅仅提供链接列表时&a…

混合密度模型GMM的似然函数(二)

设 Θ { π k , θ k } k 1 K \varTheta \{ \pi_k, \boldsymbol {\theta}_k \}_{k1}^{K} Θ{πk​,θk​}k1K​为参数向量&#xff0c; X { x 1 , ⋯ , x n } \mathcal {X} \{ {\bm x}_1, \cdots, {\bm x}_n \} X{x1​,⋯,xn​}为观测数据&#xff0c;给定数据点的独立性&a…

selenium元素定位

当我们可以打开浏览器后我们如果想要进行web测试我们自然要对网页的一些功能进行单独拿出来进行测试&#xff0c;但是我们要怎么才能拿到我们想要的元素&#xff0c;并且对其进行操作呢。 我们就以百度主页的输入框为例&#xff0c;如果我们想要王输入框中输入一些内容我们就需…

2025第十五届上海生物发酵展:江苏健达干燥盛装赴会

2025 年 8 月 7 - 9 日&#xff0c;上海新国际博览中心将迎来一场生物发酵行业的盛会 —— 第 15 届上海国际生物发酵产品与技术装备展览会&#xff08;BIOCHINA 2025&#xff09;。作为国内干燥设备领域的领军企业&#xff0c;江苏健达干燥工程有限公司受邀盛装参展&#xff0…

【效率工具】单机游戏修改方案:轻量管理器+全能平台组合

大家好&#xff01;今天我要给大家介绍两款超级实用的软件&#xff0c;专门为喜欢玩单机游戏的小伙伴们准备。 一、风灵月影管理器 不想满网翻修改器&#xff1f;这个 27M 的小工具直接帮你一键搞定&#xff0c;这款软件是由B站UP鸦无量 开发。 收录上千款游戏补丁&#xff0c;…

七天学会SpringCloud分布式微服务——01——基础概念

重点是复习体系&#xff0c;从今天6.24开始&#xff0c;确保转化为自己的东西心平气和&#xff0c;脚踏实地学习的是尚硅谷微服务 1、从单体架构到集群架构再到分布式架构 单体架构 就是 所有的功能&#xff08;服务&#xff09;模块 都部署在同一台服务器&#xff08;一台服…

三分钟学会利用deepseek将复杂信息转换成可视化图表

数据可视化是传达复杂信息的重要手段。通过将数据转化为直观的图表、图形和交互式界面,我们可以更高效地理解信息、发现趋势并做出决策。对于普通人来说,要将数据可视化可谓千难万难。但在AI工具飞速发展的今天,这个过程将会变得非常简单。今天分享的内容就是如何使用生成式…

PDF处理控件Spire.PDF系列教程:Python中快速提取PDF文本、表格、图像及文档信息

在 Python 中读取 PDF 文档是实现文档自动化、内容分析和数据提取的基础操作之一。无论你处理的是合同、报告、发票&#xff0c;还是科研论文&#xff0c;能够通过代码访问 PDF 内容&#xff0c;不仅能节省时间&#xff0c;还能带来更高效的处理流程。 要在 Python 中准确提取…

微软人工智能证书AI-102 | 如何快速通过?

微软 AI-102 考试&#xff0c;全称 “Designing and Implementing a Microsoft Azure AI Solution”&#xff0c;是微软推出的用于验证考生在 Azure 平台上设计和实施 AI 解决方案核心能力的认证考试。以下是具体介绍&#xff1a; 考试描述&#xff1a; 考试主要衡量考生实施计…

github使用指南

1、生成SSH密钥对 ssh-keygen -t ed25519 -C "你的github邮箱"然后根据提示保存路径&#xff0c;设置密码 2、将公钥添加到github cat ~/.ssh/id_ed25519.pub复制输出内容。 在gihub中点击New SSH Key&#xff0c;添加密钥 3、配置git使用SSH地址 git remote se…

AD22以上的基础操作

1.检测创建的原理图器件库 2.原理图页加大 Size&#xff1a;常规和自定义 推荐可视化栅格100mil 快捷键VG 3.原理图器件器件号排序 自动排序&#xff1a;快捷键TAA 先解锁 4.BOM(Bill of Material)物料表导出 description描述&#xff1a;类似精度。 导出各种类型bom表 5…