C语言--字符函数与字符串函数

大家好,我是残念,希望在你看完之后,能对你有所帮助,有什么不足请指正!共同学习交流
本文由:残念ing 原创CSDN首发,如需要转载请通知
个人主页:残念ing-CSDN博客,欢迎各位→点赞👍 + 收藏⭐️ + 留言📝
📣系列专栏:残念ing 的C语言系列专栏——CSDN博客

目录

前言:

1.字符分类函数

2.字符转换函数

3.strlen 函数

3.1 正确使用strlen函数

3.2 模拟实现strlen函数

4. strcpy 函数

4.1 strcpy的使用

4.2 模拟实现strcpy

5. strcat 函数

5.1 strcat的使用

5.2 模拟实现strcat

6. strcmp 函数

6.1 strcmp的使用

6.2 模拟实现strcmp

7. strncpy函数

7.1 strncpy 的使用

8.strncat 函数

8.1 strncat 的使用

9. strncmp 函数

9.1 strncmp 的使用

10. strstr 函数

10.1 strstr 的使用

10.2 模拟实现strstr

11. strtok 函数

11.1 strtok 的使用

12. strerror 函数

12.1 strerror 的使用


前言:

我们在编程过程中,我们阶乘要处理到一下字符和字符串,为了方便操作字符和字符串,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()
{int i = 0;char str[] = "Test String\n";char c;while (str[i]){c = str[i];if (islower(c))c -= 32;//小写字母转换成大写字母-32putchar(c);//打印ci++;}return 0;
}

2.字符转换函数

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

int tolower ( int c ); //将参数传进去的⼤写字⺟转⼩写 
int toupper ( int c ); //将参数传进去的⼩写字⺟转⼤写

我们知道了这两个函数,我们就可以把上面的类型写为

#include <stdio.h>
#include <ctype.h>
int main()
{int i = 0;char str[] = "Test String\n";char c;while (str[i]){c = str[i];if (islower(c))c = toupper(c);//将小写字母转换成大写字母putchar(c);i++;}return 0;
}

3.strlen 函数

size_t strlen ( const char * str );

注意:

1. 字符串以 '\0' 作为结束标志,strlen函数返回的是在字符串中 '\0' 前⾯出现的字符个数(不包 含 '\0' )。

2. 参数指向的字符串必须要以 '\0' 结束。

3. 注意函数的返回值为size_t,是⽆符号的( 易错 )

4. strlen的使⽤需要包含头⽂件(<string.h>)

3.1 正确使用strlen函数

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

3.2 模拟实现strlen函数

//方法1:
//计数器⽅式
int my_strlen(const char* str)
{int count = 0;assert(str);while (*str){count++;str++;}return count;
}
//方法2
//不能创建临时变量计数器(递归)
int my_strlen(const char* str)
{assert(str);if (*str == '\0')return 0;elsereturn 1 + my_strlen(str + 1);
}
//方法3
//指针-指针的⽅式
int my_strlen(char* str)
{assert(str);char* p = str;while (*p != '\0')p++;return p - str;
}

4. strcpy 函数

char* strcpy(char * destination, const char * source );

功能: 将源指向的C字符串复制到目标指向的数组中,包括终止空字符(并在该点停止)

注:

1. 源字符串必须以'\0'结束。

2. 会将源字符串中的'\0'拷贝到目标空间

3. 目标空间必须足够大,以确保能存放源字符串。

4. 目标空间必须可以修改

4.1 strcpy的使用

int main()
{char arr1[] = "abcdef";char arr2[20] = "";char* ret = strcpy(arr2, arr1);for (int i = 0; i < 20; i++){printf("%c", arr2[i]);}return 0;
}

4.2 模拟实现strcpy

char* my_strcpy(char* p1, const char* p2)
{while (*p2 != '\0'){*p1 = *p2;p1++;p2++;}return p1;
}
int main()
{char arr1[] = "abcdef";char arr2[20] = "";char* ret = my_strcpy(arr2, arr1);for (int i = 0; i < 20; i++){printf("%c", arr2[i]);}return 0;
}

5. strcat 函数

char* strcat(char * destination, const char * source );

 功能:将源字符串的副本追加到目标字符串。目的地的终止空字符被源的第一个字符覆盖,并且在目的地中由两个字符串串联而成的新字符串的末尾包含一个空字符。

注意:

1.源字符串必须以'\0'结束

2. 目标字符串中也得有\0,否则没办法知道追加从哪里开始

3. 目标空间必须有足够的大,能容纳下源字符串的内容

4. 目标空间必须可修改

5.1 strcat的使用

int main()
{char arr1[20] = "abcdef";char arr2[20] = "abcda";char* ret = strcat(arr1, arr2);for (int i = 0; i < 20; i++){printf("%c", arr1[i]);}return 0;
}

5.2 模拟实现strcat

char* my_strcat(char* p1, const char* p2)
{while (*p1 != '\0'){p1++;}while (*p2 != '\0'){*p1 = *p2;p1++;p2++;}*p1 = *p2;return p1;
}
int main()
{char arr1[20] = "abcdef";char arr2[20] = "abcda";char* ret = my_strcat(arr1, arr2);for (int i = 0; i < 20; i++){printf("%c", arr1[i]);}return 0;
}

6. strcmp 函数

char* strcat(const char * destination, const char * source );

功能:比较每个字符串的第一个字符。如果它们彼此相等,则继续比较下去,直到字符不同或到达终止空字符为止。

注意:

1. 第一个字符串大于第二个字符串,则返回大于0的数字

2. 第一个字符串等于第二个字符串,则返回0

3. 第一个字符串小于第二个字符串,则返回小于0的数字

比较方法:比较两个字符串中对应位置上字符ASCII码值的大小。

6.1 strcmp的使用

int main()
{char arr1[] = "abcdef";char arr2[20] = "abcda";int ret = strcmp(arr1, arr2);printf("%d\n", ret);return 0;
}

6.2 模拟实现strcmp

int my_strcmp(const char* p1, const char* p2)
{while (*p1 == *p2){if (*p1 == '\0')return 0;p1++;p2++;}if (*p1 > *p2)return 1;elsereturn -1;
}
int main()
{char arr1[] = "abcdef";char arr2[20] = "abcda";int ret = my_strcmp(arr1, arr2);printf("%d\n", ret);return 0;
}

7. strncpy函数

char * strncpy ( char * destination, const char * source, size_t num );

功能:将源字符串的第一个num字符复制到目标字符串。如果在num字符被复制之前找到了源字符串的结尾(由null字符表示),则用零填充,直到向其总共写入num字符为止。

注意:

1.拷⻉num个字符从源字符串到⽬标空间。

2. 如果源字符串的⻓度⼩于num,则拷⻉完源字符串之后,在⽬标的后边追加0,直到num个

7.1 strncpy 的使用

int main()
{char arr1[] = "world";char arr2[20] = "hello";strncpy(arr1, arr2, 3);//当超过字符串长度时,添加'\0'printf("%s", arr1);return 0;
}

8.strncat 函数

char * strncat ( char * destination, const char * source, size_t num );

功能:将source指向字符串的前num个字符追加到destination指向的字符串末尾,再追加⼀个 \0 字符。如果source 指向的字符串的⻓度小于于num的时候,只会将字符串中到 \0 的内容追加到destination指向的字符串末尾。

8.1 strncat 的使用

int main()
{char arr1[20] = "world";char arr2[20] = "hello";strncat(arr1, arr2, 3);//会追加\0printf("%s", arr1);return 0;
}

9. strncmp 函数

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

功能:⽐较str1和str2的前num个字符,如果相等就继续往后⽐较,最多⽐较num个字⺟,如果提前发现不一样,就提前结束,⼤的字符所在的字符串⼤于另外⼀个。如果num个字符都相等,就是相等返回0。

9.1 strncmp 的使用

int main()
{char arr1[20] = "world";char arr2[20] = "hello";int ret = strncmp(arr1, arr2, 3);//会追加\0printf("%d", ret);return 0;
}

10. strstr 函数

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

功能:函数返回字符串str2在字符串str1中第⼀次出现的位置,字符 串的⽐较匹配不包含 \0 字符,以 \0 作为结束标志。

10.1 strstr 的使用

int main()
{char arr1[] = "i think you ";char arr2[] = "think";char* ret = strstr(arr1, arr2);if (ret != NULL)printf("%s\n", ret);elseprintf("没有找到\n");return 0;
}

10.2 模拟实现strstr

char* my_strstr(char* p1, char* p2)
{if (*p2 == '\0'){return NULL;}char* cur = p1;while (*cur != '\0'){char* p3 = cur;char* p4 = p2;while (*p3 == *p4){p3++;p4++;}if (*p4 == '\0'){return p2;}cur++;}return NULL;
}
int main()
{char arr1[] = "I think you ";char arr2[] = "think";char* ret = my_strstr(arr1, arr2);if (ret != NULL)printf("%s\n", ret);elseprintf("没有找到\n");return 0;
}

11. strtok 函数

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

注意:

1.sep参数指向⼀个字符串,定义了⽤作分隔符的字符集合

2. 第⼀个参数指定⼀个字符串,它包含了0个或者多个由sep字符串中⼀个或者多个分隔符分割的标 记。

3.strtok函数找到str中的下⼀个标记,并将其⽤ \0 结尾,返回⼀个指向这个标记的指针。(注: strtok函数会改变被操作的字符串,所以在使⽤strtok函数切分的字符串⼀般都是临时拷⻉的内容 并且可修改。)

4. strtok函数的第⼀个参数不为 NULL ,函数将找到str中第⼀个标记,strtok函数将保存它在字符串 中的位置。

5. strtok函数的第⼀个参数为 NULL ,函数将在同⼀个字符串中被保存的位置开始,查找下⼀个标 记。

6. 如果字符串中不存在更多的标记,则返回 NULL 指针。

11.1 strtok 的使用

int main()
{char arr[] = "192.168.6.111";char* sep = " .";char* str = NULL;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函数就可以将错误对应的错误信息字符串的地址返回。

12.1 strerror 的使用

#include <errno.h>
#include <string.h>
#include <stdio.h>
//我们打印⼀下0~10这些错误码对应的信息
int main()
{int i = 0;for (i = 0; i <= 10; i++) {printf("%s\n", strerror(i));}return 0;
}

打印结果:

补充:与这个函数功能比较像的函数还有perror函数,prrror函数是直接将错误信息打印出来。perror函数打印完参数部分的字符串后,再打印⼀个冒号和⼀个空格,再打印错误信息。

#include <stdio.h>
#include <string.h>
#include <errno.h>
int main()
{FILE* pFile;pFile = fopen("unexist.ent", "r");if (pFile == NULL)perror("Error opening file unexist.ent");return 0;
}

打印结果:

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

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

相关文章

推荐一款好用的PDF阅读器

下载地址: https://download.csdn.net/download/a876106354/88643909

翻译: LLMs大语言模型影响到高工资的的白领知识工作者 加速各行各业的自动化潜力 Automation potential across sectors

我们已经探讨了生成人工智能可能对您的工作有用&#xff0c;也讨论了分析其对企业的影响。现在&#xff0c;让我们拉远镜头&#xff0c;看看它对不同公司的工作角色以及对不同行业部门的影响。这个视频的结果对特定企业可能不那么直接可行&#xff0c;但也许这会帮助您思考并尝…

Node.js使用Express框架写服务端接口时,如何将接口拆分到不同文件中

项目目录结构说明&#xff1a; node.js连接mysql数据库步骤可参考&#xff1a;Node.js 连接 MySQL | 菜鸟教程 1、拆分之前的写法&#xff0c;未区分模块&#xff0c;所有接口api都写在了入口文件app.js中&#xff1b; 需求&#xff1a;想要将接口api拆分成根据不同的业务模块…

在4*4的平面上计算2a1+1+1

0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 1 在4*4的平面上有2个点&#xff0c;保持2a1的结构&#xff0c;然后向剩余的14个格子里随机扔2个石子。 共有14*13/291种可能 1 - - - 2 - - - 3 - - 1 4 - - - 1 1 - 1 1 - - - - - - - 1 - - …

深度剖析知识图谱:方法、工具与实战案例

&#x1f482; 个人网站:【 海拥】【神级代码资源网站】【办公神器】&#x1f91f; 基于Web端打造的&#xff1a;&#x1f449;轻量化工具创作平台&#x1f485; 想寻找共同学习交流的小伙伴&#xff0c;请点击【全栈技术交流群】 知识图谱作为一种强大的知识表示和关联技术&am…

非递归实现的快速排序

目录 序列文章 前言 学前补充 非递归快速排序 注意事项&#xff08;重要&#xff09; 实现步骤 代码实现 时空复杂度 快速排序的特性 栈的相关代码 序列文章 非递归实现的快速排序&#xff1a;http://t.csdnimg.cn/UEcL6 快速排序的挖坑法与双指针法&#xff1a;ht…

创建自定义 gym env 教程

gym-0.26.1 pygame-2.1.2 自定义环境 GridWolrdEnv 教程参考 官网自定义环境 &#xff0c;我把一些可能有疑惑的地方讲解下。 首先整体文件结构, 这里省略了wrappers gym-examples/main.py # 这个是测试自定义的环境setup.py gym_examples/__init__.pyenvs/__init__.pygri…

【NI-RIO入门】扫描模式

于NI KB摘录 所有CompactRIO设备都可以访问CompactRIO扫描引擎和LabVIEW FPGA。 CompactRIO 904x 系列是第一个引入 DAQmx 功能的产品线。 扫描引擎&#xff08;IO 变量&#xff09; – 主要为迁移和初始开发而设计。控制循环频率高达 1 kHz1&#xff0c;性能控制器上的频率更…

【ArkTS】路由传参

传参 使用router.pushUrl()&#xff0c;router.push()官方不推荐再使用了。 格式&#xff1a; router.pushUrl({url: 路由地址,params:{参数名&#xff1a;值} )跳转时需要注意路由表中是否包含路由地址。 路由表路径&#xff1a; entry > src > main > resources &g…

专栏十六:bulk以及单细胞空转中的progeny通路分析

progeny本身有自己的R包,可以提取通路基因集信息,团队把他嵌入另一个R包decoupleR中完成富集分析。decoupleR自己有详细的针对bulk和scRNAseq的教程 简单安装一下 devtools::install_github(saezlab/OmnipathR) devtools::install_github("saezlab/progeny") Bio…

Sectigo DV多域名证书能保护几个域名

多域名SSL证书不限制受保护的域名的类型&#xff0c;可以时多个主域名或者子域名&#xff0c;多域名SSL证书都可以同时保护&#xff0c;比较灵活。但是&#xff0c;多域名https证书并不是免费无限制保护域名数量&#xff0c;一把的多域名SSL证书默认保护3-5个域名记录&#xff…

LVS+Keepalived 高可用集群

一.Keepalived工具介绍 1.支持故障自动切换(Failover) 2.支持节点健康状态检查(Health Checking) 3.基于vrrp协议完成地址流动 4.为vip地址所在的节点生成ipvs规则(在配置文件中预先定义) 5.为ipvs集群的各RS做健康状态检测 6.基于脚本调用接口完成脚本中定义的功能&…