C语言字符函数和字符串函数详解

       Hello, 大家好,我是一代,今天给大家带来有关字符函数和字符串函数的有关知识

       所属专栏:C语言

       创作不易,望得到各位佬们的互三呦

一.字符函数

在C语言中有一些函数是专门为字符设计的,这些函数的使用都需要包含一个头文件<ctype.h>

如:(注:以下函数原型都很相似,都为int 函数(int 字符))

函数

iscntrl                       检查字符是否为可控制字符

isspace                    检查字符是否为一个空格、制表符、换行,换页’\f‘或回车

isdigit                       检查字符是否为数字’0‘-’9‘

isxdigit                     检查字符是否为16进制的字符包括所有10进制数字,小写字母,a-f,大                                  写字符A-F

islower                     检查字符是否为小写字母

isupper                    检查字符是否为大写字母

isalpha                     检查字符是否为可打印字符

isalnum                    检查字符是否为字母或十进制数字

ispunct                     检查字符是否为可打印标点字符(不属于数字和字母的图形字符)

isgraph                     检查字符是否为可打印字符(不包括空格)

isprint                       检查字符是否为可打印字符(包括空格)

以上函数当条件满足时的返回值都为非0的整数

如: islower(c),如果c是小写字母,返回非零整数,不是小写字母就返回0

#include<stdio.h>
#include<ctype.h>
int main()
{printf("%d", islower('b'));
}

二.字符转换函数

C语言提供了两种字符转换函数

1.int tolower(int c)    将传进去的大写字母转换为小写

2. int upper(int c)    将传进去的小写字母转换为大写

如:

#include<stdio.h>
#include<ctype.h>
int main()
{printf("%c", tolower('A'));//将大写字母转小写,如果不为大写字母则不改变
}

 三.字符串函数及字符串模拟

这里字符串函数包含的头文件都为string.h

一.strlen函数

函数原型:size_t strlen (const char*str)

头文件:string.h

功能:计算字符串长度(字符串以‘\0’为结束标志,strlen函数返回的是在字符串中‘\0’前面面出现的字符个数)

注意:1.参数指向的字符串必须以'\0'为结束标志,否则不知道字符串在哪里结束,返回随机值

2.函数的返回值为size_t , 是无符号的(下面会举相关例题)

就strlen的返回值有一个例题:

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

这里很多人会认为输出的是<,但实际输出的是>,为什么呢?其实是以为strlen的返回值为size_t,其表示的数都为无符号数,也就是表示的为正数,比如-1,因为返回类型为size_t,其最高位的符号为表示的就为2的32次方,而不是表示正负号。

strlen函数模拟实现

方法1:指针减指针,指针减指针表示的是两个指针之间相差的元素个数

​
#include<stdio.h>
#include<assert.h>
size_t my_strlen(char*str)
{assert(str);//防止str为空指针,包含头文件为assert.hchar* p = str;while (*str != '\0'){str++;}return str - p;
}
int main()
{const char* str1 = "acbdh";printf("%zd", my_strlen(str1));return 0;
}​

 方法2:计数器方式,用count记录元素个数

#include<stdio.h>
#include<assert.h>
size_t my_strlen(const char* str)
{assert(str);//防止str为空指针,包含头文件为assert.hint count = 0;while (*str != '\0'){str++;count++;}return count;
}
int main()
{const char* str1 = "acbdh";printf("%zd", my_strlen(str1));return 0;
}

方法3:递归方式

#include<stdio.h>
#include<assert.h>
size_t my_strlen(const char* str)
{assert(str);//防止str为空指针,包含头文件为assert.hif (*str == '\0')return 0;return my_strlen(str + 1) + 1;
}
int main()
{const char* str1 = "acbdh";printf("%zd", my_strlen(str1));return 0;
}

二.strcpy函数

函数原型:char* strcpy(char* destination ,const char* source)

功能:把源字符串复制到目标字符串

注意:1.源字符必须以‘\0’结束

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

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

4.目标空间必须可修改

strcpy的模拟实现

​
#include<stdio.h>
#include<assert.h>
char* my_strcpy(char* str1, char* str2)
{assert(str1 && str2);char* s = str1;while (*str1++ = *str2++);//先解引用在++return s;
}
int main()
{char str1[] = "acbdh";char* str2 = "acb";printf("%s", my_strcpy(str1,str2));return 0;
}​

三.strcat函数

函数原型:char* strcat(char*dest,const char*str)

功能:将两个字符串拼接起来,将str拼接到dest上

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

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

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

4.目标空间必须可修改

strcat模拟实现

#include<stdio.h>
#include<assert.h>
char* my_strcat(char* str1,const char*str2)
{assert(str1&&str2);//防止str为空指针,包含头文件为assert.hchar* s = str1;while(*str1){str1++;}while (*str1++ = *str2++);//当*str2为'\0'时表达式结果为假return s;}
int main()
{char str1[20] = "acbdh";const char* str2 = "sad";printf("%s", my_strcat(str1,str2));return 0;
}

四.strcmp函数

函数原型:int strcmp(char*str1,char*str2)

功能:比较两个字符串的大小,若第一个字符串大于第二个字符串,则返回大于0的数字

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

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

注意:比较两个字符串大小是用ascii值比较,一个一个字符比较

strcmp函数模拟

#include<stdio.h>
#include<assert.h>
int  my_strcmp(char* str1,const char*str2)
{assert(str1&&str2);//防止str为空指针,包含头文件为assert.hwhile (*str1 == *str2){if (*str1 == '\0')//代表*str1和*str2都为'\0'return 0;str1++;str2++;}return *str1 - *str2;
}
int main()
{const char *str1 = "acbdh";const char* str2 = "sad";printf("%d", my_strcmp(str1,str2));return 0;
}

 五.strncpy函数

函数原型:char* strncpy(char* destination,char *source,size_t num)

功能:拷贝num个字符从源字符串到目标空间

注意:如果源字符串的长度小于num,则拷贝完源字符后,到目标空间的后边追加'\0',直至num个

strncpy函数模拟

#include <stdio.h>
#include <string.h>
#include<assert.h>
char* my_strncpy(char* str1, const char* str2, size_t n)
{assert(str1 && str2);char* s = str1;for(int i=0;i<n && str2!='\0';i++){*str1++ = *str2++;}*str1 = '\0';return s;
}
int main()
{char str1[20]="name ";char str2[20]="str";printf("%s", my_strncpy(str1, str2, 1));
}

六.strncat函数

函数原型:char*strncat(char* destination ,const char*source ,size_t num)

功能 :将source指向字符串的前num个字符追加到destination指向的字符串末尾,再追加⼀个 \0 字符

注意:如果source 指向的字符串的长度小于num的时候,只会将字符串中到'\0'的内容追加到destination指向的字符串末尾

strncat模拟

#include<stdio.h>
#include<assert.h>
char* my_strncat(char* dest, char* str, size_t n)
{assert(dest && str);int i;char* s = dest;while (*dest != '\0'){dest++;}for (i = 0; i < n && str[i] != '\0'; i++){dest[i] = str[i];}dest[i] = '\0';return s;
}
int main()
{char str[] = { "abcd" };char dest[20] = { "mnkj***\0***" };char* ret = my_strncat(dest, str, 2);printf("%s", ret);
}

七.strncmp函数

函数原型:(const char*str1, const char*str2,size_t num)

功能: 比较str1和str2的前num个字符,如果相等就继续往后⽐较,最多比较num个字⺟,如果提前发现不一样,就提前结束。如果num个字符都相等,就是相等返回0,如果str字符串<str2字符串,就返回小于0的数,反之,返回大于0的数

strncmp函数模拟

#include<stdio.h>
#include<assert.h>
int  my_strncmp(const char* str1, const char* str2,size_t num)
{assert(str1 && str2);//防止str为空指针,包含头文件为assert.hwhile (*str1 == *str2&& num--){if (*str1 == '\0')//代表*str1和*str2都为'\0'return 0;str1++;str2++;}if (num == 0){return 0;}return *str1 - *str2;
}
int main()
{const char* str1 = "afbdh";const char* str2 = "afbh";printf("%d", my_strncmp(str1, str2, 9));return 0;
}

八.strstr函数

函数原型:char* strstr(const char *str1,const char*str2)

功能: 返回字符串str2在字符串str1中第⼀次出现的位置

strstr函数模拟

#include<stdio.h>
#include<assert.h>
char* my_strstr(const char* str1, const char* str2)
{assert(str1 && str2);const char* p1 = NULL;const char* p2 = NULL;const char* cur = str1;if (*str2 == '\0')return str1;while (*cur){p1 = cur;//记录str1开始的起始位置p2 = str2;//记录str2的位置while (*p1 == *p2 && *p1 != '\0' && *p2 != '\0'){p1++;p2++;}if (p2 == '\0'){return (char*)cur;//将const char*强转}if (*p1 == '\0'){return NULL;}cur++;//起始位置不可以,刷新起始位置}return NULL;}
int main()
{char a[] = { "abbbcde" };char b[] = { "bc" };printf("%s", my_strstr(a, b));
}

 九.strtok函数

函数原型:char* strtok(char*str1, const char*str2)

功能: 将str1进行分割,str2中包含分割符(第⼀个参数指定⼀个字符串,它包含了0个或者多个由str2字符串中⼀个或者多个分隔符分割的标记)

注意:1.strtok函数找到str1中出现的分割符,将其标记,并将其之前的字符串⽤ \0 结尾,返回⼀个指向这个标记的指针,下次用的时候第一个参数为NULL。(注: strtok函数会改变被操作的字符串,所以在使⽤strtok函数切分的字符串⼀般都是临时拷贝的内容 并且可修改。)

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

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

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

strstr使用示例

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

九.strerror函数

函数原型:char*strerror (int errnum)

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

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

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

相关文章

管理类联考–复试–英文面试–问题–WhatWhyHow--纯英文汇总版

文章目录 Do you have any hobbies? What are you interested in? What do you usually do in your spare time? Could you tell me something about your family&#xff1f; Could you briefly introduce your family? What is your hometown like? Please tell me so…

在Arm 虚拟硬件(AVH)部署深度学习OCR算法

AI算法的嵌入式部署 AI算法在独立的设备上运行其实就是行业内的嵌入式AI的概念, 大致过程如下: 开发AI模型, 2.对数据集进行处理, 3.训练AI模型并验证效果, 4.转成ONNX格式(ONNX:万金油中间格式,给模型优化和部署带来了更多可能性)或者借助libtorch或者TensorFlow来部署C++版…

Prometheus+Grafana 监控Tongweb7

文章目录 1.准备工作2.Tongweb7部署3.Prometheus部署4.上传jar包并配置Tongweb75.Prometheus配置6.安装和配置Grafana 1.准备工作 本次参考&#xff1a;Prometheus监控Tongweb容器 1.使用虚拟机ip&#xff1a;192.168.10.51&#xff08;tongweb&#xff09;&#xff0c;192.1…

day14-SpringBoot 原理篇

一、配置优先级 SpringBoot 中支持三种格式的配置文件&#xff1a; 注意事项 虽然 springboot 支持多种格式配置文件&#xff0c;但是在项目开发时&#xff0c;推荐统一使用一种格式的配置 &#xff08;yml 是主流&#xff09;。 配置文件优先级排名&#xff08;从高到低&…

【设计模式】Java 设计模式之模板命令模式(Command)

命令模式&#xff08;Command&#xff09;的深入分析与实战解读 一、概述 命令模式是一种将请求封装为对象从而使你可用不同的请求把客户端与接受请求的对象解耦的模式。在命令模式中&#xff0c;命令对象使得发送者与接收者之间解耦&#xff0c;发送者通过命令对象来执行请求…

嵌套循环实现九九乘法表

大家好&#xff1a; 衷心希望各位点赞。 您的问题请留在评论区&#xff0c;我会及时回答。 案例描述 利用嵌套循环&#xff0c;实现九九乘法表。 代码 #include <iostream> #include <Windows.h>using namespace std;int main(void) {//外层循环执行一次&#…

v-bind 绑定 class 与 style 基础用法

使用 v-bind 指令绑定 class 和 style 时语法相对复杂一些&#xff0c;这两者是可以互相替代的&#xff0c;均用于响应更新HTML元素的属性&#xff0c; v-bind 绑定 class 属性可以改写成绑定 style 属性&#xff0c;只是 css 属性位置变了而已。 1. 绑定 class 属性 1.1 数组…

学了 Python 但又感觉没学 Python 不如重学 Python - day4(数据类型:列表)

目录 1、创建列表 2、列表基本操作 3、索引与分片 4、矩阵 5、常用列表方法 &#xff08;1&#xff09;append 与 extend 方法 &#xff08;2&#xff09;insert 方法 &#xff08;3&#xff09;remove 与 pop 方法 &#xff08;4&#xff09;del 语句与 clear 方法 …

conda创建环境网络报错解决办法

文章目录 一、报错示例&#xff1a;二、解决办法&#xff1a;2.1 查看配置 conda config --show-sources2.2 修改文件 /home/XXXX/.condarc 一、报错示例&#xff1a; UnavailableInvalidChannel: HTTP 404 NOT FOUND for channel nvidia <http://mirrors.tuna.tsinghua.ed…

【机器学习-05】模型的评估与选择

在前面【机器学习-01】机器学习基本概念与建模流程的文章中我们已经知道了机器学习的一些基本概念和模型构建的流程&#xff0c;本章我们将介绍模型训练出来后如何对模型进行评估和选择等 1、 误差与过拟合 学习器对样本的实际预测结果与真实值之间的差异&#xff0c;我们称之…

一次完整的 HTTP 请求所经历的步骤

1&#xff1a; DNS 解析(通过访问的域名找出其 IP 地址&#xff0c;递归搜索)。 2&#xff1a; HTTP 请求&#xff0c;当输入一个请求时&#xff0c;建立一个 Socket 连接发起 TCP的 3 次握手。如果是 HTTPS 请求&#xff0c;会略微有不同。 3&#xff1a; 客户端向服务器发…

Redis 搭建主从集群

文章目录 1. 主从集群架构1.1 准备实例和配置1.2 启动1.3 开启主从关系1.4 测试 2. 主从同步原理2.1 全量同步2.2 增量同步repl_backlog原理 2.3 主从同步优化小结 单节点的 Redis 并发能力有限&#xff0c;要进一步提高 Redis 的并发能力&#xff0c;就需要搭建主从集群&#…

杰发科技AC7801——Flash数据读取

0. 简介 因为需要对Flash做CRC校验&#xff0c;第一步先把flash数据读出来。 1. 代码 代码如下所示 #include "ac780x_eflash.h" #include "string.h" #define TestSize 1024 ///< 4K #define TestAddressStart 0x08000000 uint8_t Data[7000]; int…

<DFS剪枝>数字王国之军训排队

DFS剪枝 其实就是将搜索过程一些不必要的部分直接剔除掉。 剪枝是回溯法的一种重要优化手段&#xff0c;往往需要先写一个暴力搜索&#xff0c;然后找到某些特殊的数学关系&#xff0c;或者逻辑关系&#xff0c;通过它们的约束让搜索树尽可能浅而小&#xff0c;从而达到降低时间…

MAC地址(静态、黑洞、优先级)

拓扑图 配置 1&#xff09;静态MAC地址配置 mac-address learning disable命令用来关闭MAC地址学习功能。 关闭MAC地址学习功能后&#xff0c;设备将不会再从该接口学习新的MAC地址。关闭MAC地址学习后可配置的动作有discard和forward。 关闭MAC地址学习功能的缺省动作为fo…

NSS [SWPUCTF 2022 新生赛]ez_ez_unserialize

NSS [SWPUCTF 2022 新生赛]ez_ez_unserialize 开题&#xff0c;直接给了题目源码。 简单看了一下&#xff0c;题目告诉我们flag在哪&#xff0c;而且类中有高亮文件方法。怎么拿flag已经很明显了。关键点在于__weakup()魔术方法固定死了我们高亮的文件。所以这题只需要绕过__w…

【故障排查】10分钟解决Quartz重复调度的疑难杂症

我司使用Apache DolphinScheduler作为调度框架很久了&#xff0c;感兴趣的小伙伴可以看看这些干货文章&#xff1a; 因为之前监控到会出现重复的调度的问题&#xff0c;所以此文记录排查重复调度问题的全过程&#xff0c;希望对社区其他的小伙伴能够起到抛砖引玉的作用&#x…

【学习】python函数语法(面像对象、封装函数)

阅读开源深度学习源码的时候&#xff0c;使用到了很多封装函数以及Python的高级语法&#xff0c;看起来很混乱很痛苦很困难。对python函数语法做个总结&#xff01;&#xff01;&#xff01; Table of Contents 熟练Python语法&#xff0c;尤其是函数参数、迭代器与生成器、函…

使用Redis做缓存的小案例

如果不了解Redis&#xff0c;可以查看本人博客&#xff1a;Redis入门 Redis基于内存&#xff0c;因此查询速度快&#xff0c;常常可以用来作为缓存使用&#xff0c;缓存就是我们在内存中开辟一段区域来存储我们查询比较频繁的数据&#xff0c;这样&#xff0c;我们在下一次查询…

C#,图论与图算法,有向图(Directed Graph)的环(Cycle)的普通判断算法与源代码

1 检查该图是否包含循环 给定一个有向图,检查该图是否包含循环。如果给定的图形至少包含一个循环,则函数应返回true,否则返回false。 方法:深度优先遍历可用于检测图中的循环。连接图的DFS生成树。只有当图中存在后缘时,图中才存在循环。后边是从节点到自身(自循环)或…