【我爱C语言】详解字符函数isdigit和字符串转换函数(atoi和snprintf实现互相转换字符串)三种strlen模拟实现

news2024/11/24 9:07:52
  🌈write in front :

🔍个人主页 : @啊森要自信的主页

✏️真正相信奇迹的家伙,本身和奇迹一样了不起啊!

欢迎大家关注🔍点赞👍收藏⭐️留言📝>希望看完我的文章对你有小小的帮助,如有错误,可以指出,让我们一起探讨学习交流,一起加油鸭。 请添加图片描述

文章目录

  • ✒️ 前言
  • 💯字符分类函数
  • 💯 💯字符串转换函数
  • 💯 💯 💯strlen的使⽤
  • 💯 💯 💯 💯 strlen的3种模拟实现方式
  • 📝总结


✒️ 前言

本小节我们将学习字符分类函数,字符串转换函数,使用库函数实现函数(tolower,toupper) 的大小写转换,当然还有字符串转换成整数(isdigit和atoi),使用(snprintf)将整数怎么转换回字符串。最后还有strlen函数的三种模拟实现。文章干货满满,让我们学习起来!


💯字符分类函数

这些函数都定义在 ctype.h 头文件中。
它们的参数 c 是字符,返回值为非零(true)零(false)
ctype.h --iscntrl©- 任何控制字符
- isalpha© - 检查是否为字母字符( a ~zA ~ Z)
-isupper© - 检查是否为大写字母字符(A ~Z)
- islower© - 检查是否为小写字母字符(A ~Z)
- isdigit© - 检查是否为数字字符(十进制 0 ~ 9)
- isxdigit© - 检查是否为十六进制数字字符 (包括所有进制数字,小写字母a ~ f ,大写字母A ~ F
- isalnum© - 检查是否为字母或数字字符( a ~zA ~ Z0 ~ 9)
- isspace© - 检查是否为空白字符(空白字符:空格‘’、换页‘\f’,换行‘\n’,制表符‘\t’或者垂直制表符‘\v’等)
- ispunct© - 检查是否为标点符号字符(任何不属于数字或者字母的图形字符(可打印))
- isprint© - 检查是否为可打印字符(任何可打印字符,包括图形字符和空白字符)
- isgraph© - 检查是否为除空格外的可打印字符(任何图形字符)

这些函数的使⽤⽅法⾮常类似,如 islower:

 int islower ( int c );

islower 是能够判断参数部分的 c 是否是⼩写字⺟的。
通过返回值来说明是否是⼩写字⺟,如果是⼩写字⺟就返回⾮0的整数,如果不是⼩写字⺟,则返回0。

#include <ctype.h>

int main()
{
	char arr[] = "A SeN yaO Zi XIn";
	int i = 0;
	while (arr[i])
	{
		if (islower(arr[i]))
		{
			arr[i] = arr[i] - 32;
		}
		putchar(arr[i]);
		i++;
	}
	return 0;
}

在这里插入图片描述

例如isalpha

#include <ctype.h>

int main() 
{
	char c = 'a';

	if (isalpha(c))//检查是否为==字母==字符( 'a '~'z' 或'A' ~ 'Z')
	{
		printf("%c is an alphabetic character.\n", c);
	}

	return 0;
}

💯 💯字符串转换函数

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

这两个函数用于转换字母字符的大小写。

例如:

#include <ctype.h>
#include <stdio.h>
int main()
{
	char arr[] = "I Am A Student";
	int i = 0;
	while (arr[i])
	{
		if (isupper(arr[i]))
		{
			//arr[i] = arr[i] + 32;
			arr[i] = tolower(arr[i]);
		}
		putchar(arr[i]);
		i++;
	}
	return 0;
}

既然可以字符大小写换,那有没有字符串和数字转换的函数呢?答案是有的。

  • isdigit检查单个字符是否是数字
  • atoi将整个字符串转换为整数
  1. isdigit函数:
int isdigit(int c);

isdigit函数用于检查给定字符是否是一个ASCII数字字符。它返回一个非零值或0来指示字符是否是数字(在ctype.h头文件中)。

  1. atoi函数:
int atoi(const char *str);

atoi函数用于将字符串转换为相应的整数值。它会跳过字符串前面的空格,然后将字符串中连续的数字字符转换为整数返回。(使用atoi函数需要包含stdlib.h头文件)

它们都是标准C库中常用的字符串和数字转换函数。

isdigit:用于检查单个字符c是否是一个数字字符。如果是数字字符,它会返回c对应的ASCII数字值,否则返回0
atoi(str):用于将字符串str转换为整数。它会跳过字符串前面的空白字符,然后将字符串中的数字字符转换为相应的整数值返回。

这两个函数经常一起使用,来实现字符串到整数的转换:

#include <stdio.h>
#include <stdlib.h>
#include <ctype.h>
int main() 
{
	char str[] = "123";

	int i = 0;
	int sum = 0;
	for (i = 0; str[i] != '\0'; i++) 
	{
		if (isdigit(str[i])) //使用isdigit检查每个字符是否是数字
		{
			sum = sum * 10 + (str[i] - '0');
		}//如果是数字,则取它的ASCII值减去'0'得到实际值,累加计算字符串整数值
	}

	printf("isdigit 逐个转换: %d\n", sum);

	int num = atoi(str);//或者直接使用atoi直接转换整个字符串
	printf("atoi 直接转换: %d\n", num);

	return 0;
}

运行代码如下:
在这里插入图片描述
既然有实现字符串到整数的转换,那也可以将整数转换为字符串吧!没错!老铁,杠杠的!

snprintf函数可以将整数转换为字符串

snprintf函数原型:

int snprintf ( char * str, size_t n, const char * format, ... );

其中:

  • char *str 是目标字符串缓冲区指针
  • size_t size 是缓冲区大小缓冲区中要使用的最大字节数。生成的字符串的长度最多为 n-1,为额外的终止 null 字符留出空间。(size_t 是无符号整数类型。)
  • const char *format 是格式字符串
  • ... 表示可变参数列表,根据格式字符串,该函数可能需要一系列附加参数,每个参数都包含一个值,用于替换格式字符串中的格式说明符(或指向存储位置的指针,对于 n)。这些参数的数量应至少与格式说明符中指定的值数一样多。该函数将忽略其他参数。

注意:snprintf函数的返回类型是一个int值,表示实际写入目标字符串(不包括终止null字符’\0’)的字符数。
如果返回值等于或大于指定的size,就表示格式化字符串写入目标缓冲区时会发生截断。
如果足够大,则将写入的字符数,不包括终止 null 字符。
如果发生编码错误,则返回负数。
请注意,只有当此返回值为非负且小于 时,字符串才被完全写入。

总结:我们可以把可变参数(...)设置为整数num,(const char * format)格式是==“%d”,存储进大小为size_t n目标字符数组str==中就可以解决了。

上代码:

#include <stdio.h>

int main()
{
    int num = 123456;//定义一个整数num
    char str[10] ;//字符数组str作为目标缓冲区

    int len = snprintf(str, sizeof(str), "%d", num);
				//调用snprintf进行格式化转换,并用len接收返回值
	printf("%d\n", len);//查看他返回的写入字符串的个数

    if (len < 0) 
    {
        printf("编码错误\n");
        return -1;
    }

    if (len >= sizeof(str)) 
    {
        printf("截断,数字长度大于缓冲区大小\n");
    }
    else
    {
        printf("字符串转换成功: %s\n", str);
        printf("字符串写入成功且第二个元素是: %c\n", str[1]);

    }

    return 0;
}

代码运行:
在这里插入图片描述

在代码中,我没有给字符数组str赋值 char str[10] ;//字符数组str作为目标缓冲区,在整数123456调试中我们可以看到str[6]=='\0'
'\0'哪里来的呢?
我们通过前面知道snprintf函数的返回值表示实际写入目标字符串的字符数,但不包括结尾的null字符’\0’。
至于null字符'\0'是怎么来的,snprintf在写入字符串时,会自动在结尾添加一个null字符'\0’,用来标识字符串的结束。
整数"123456"转换为字符串,需要6个字符加1个null字符,总长度是7
因此我们需要在目标字符串内存中预留了null字符所占的空间
在这里插入图片描述

	int num = 1234567890;//将num重新定义10个整数
    char str[10] ;//字符数组str作为目标缓冲区

因此我们需要在目标字符串内存中预留了null字符所占的空间,不然会发生截断
在这里插入图片描述

💯 💯 💯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()
{
	char arr1[] = "abcdef";//[a b c d e f \0]
	char arr2[] = { 'a', 'b', 'c' ,'\0'};//[a b c]
	size_t len = strlen(arr2);
	printf("%zd\n", len);

	return 0;
}

有个易错题:

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

答案是什么呢?
运行启动:
在这里插入图片描述
为什么呢?

strlen返回的字符串长度类型是size_t,它是一个无符号整数类型。

  • str1长度为6
  • str2长度为3
  • strlen(str2) - strlen(str1) 计算为3 - 6,结果是-3
  • 但是-3作为size_t类型,它是一个无符号整数,所以它的值实际上是大于0的,所以打印的是"str2>str1"

注意:
size_t是一个无符号整数类型
例如在32位系统中:
size_t最大值为2^32 - 1
-3作为size_t,它的值就是2^32 - 1 - 3他的值远远大于0.

💯 💯 💯 💯 strlen的3种模拟实现方式

  1. 创建临时变量count
#include <stdio.h>
#include <string.h>
#include <assert.h>

size_t my_strlen1(const char* str)
{
	size_t count = 0;
	assert(str != NULL);
	while (*str)
	{
		count++;
		str++;
	}
	return count;
}

2.指针减指针

start指针记录起始位置,str指针遍历字符串,返回二者差值即为长度。

size_t my_strlen2(const char* str)
{
	assert(str);
	const char* start = str;
	while (*str)
	{
		str++;
	}
	return str - start;//两指针相减得到的是中间的元素个数
}

3.函数递归遍历(不使用临时变量,求字符串长度)

  size_t my_strlen(const char* str)
{
	if (*str == '\0')
		return 0;
	else
		return 1 + my_strlen(str + 1);
}

int main()
{
	char arr[] = "abcdef";
	size_t len = my_strlen(arr);
	printf("%zd\n", len);

	return 0;
}
例如,递归abcdef

my_strlen("abcdef");
1+my_strlen("bcdef");
1+1+my_strlen("cdef");
1+1+1+my_strlen("def");
1+1+1+1+my_strlen("ef");
1+1+1+1+1+my_strlen("f");
1+1+1+1+1+1+my_strlen("");
1+1+1+1+1+1+0;

📝总结

通过学习字符分类函数、字符串转换函数和字符串长度计算的原理和用法,我们可以更好地理解字符和字符串的处理方式,并能够灵活运用这些函数进行字符和字符串的处理。这些函数在实际的开发中经常会用到,掌握它们的使用方法对于提高开发效率和代码质量都很重要。

感谢你的收看,如果文章有错误,可以指出,我不胜感激,让我们一起学习交流,如果文章可以给你一个小小帮助,可以给博主点一个小小的赞😘
请添加图片描述

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

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

相关文章

使用Python实现轮盘赌选择法Roulette Wheel Selection Method in Python

一、引言 最近在手写遗传算法&#xff0c;想尝试解决一些优化问题。然而&#xff0c;在编码的过程中&#xff0c;自己发现了很多都不懂的问题。比如&#xff0c;交叉的操作&#xff0c;有单点交叉、两点交叉和多点交叉&#xff0c;具体选哪一种会更好呢&#xff1f;未知。还有交…

异常检测 | 基于孤立森林(Isolation Forest)的数据异常数据检测(结合t-SNE降维可视化)

异常检测 | MATLAB实现基于孤立森林的数据异常检测 目录 异常检测 | MATLAB实现基于孤立森林的数据异常检测效果一览基本介绍程序设计参考资料 效果一览 基本介绍 Matlab实现基于孤立森林(Isolation Forest)的数据异常数据检测可视化&#xff08;完整源码和数据) 基于孤立森林(…

好用免费的AI换脸5个工具

在当今社会的发展中&#xff0c;人工智能&#xff08;Artificial Intelligence, AI&#xff09;扮演着关键的角色&#xff0c;其应用领域不断扩展。作为AI的一个分支&#xff0c;换脸技术近年来备受欢迎。这项技术使得将一个人的面部特征迁移到另一个人的照片或视频成为可能。除…

【React】路由的基础使用

react-router-dom6的基础使用 1、安装依赖 npm i react-router-dom默认安装最新版本的 2、在src/router/index.js import { createBrowserRouter } from "react-router-dom"/* createBrowserRouter&#xff1a;[/home]--h5路由createHashRouter&#xff1a;[/#/ho…

各种滤波算法的比较(GF、KF、EKF、UKF、PF),内附简单实现代码

目录 一、前言 二、滤波算法介绍 1、GF&#xff08;高斯滤波&#xff09; 2、KF&#xff08;卡尔曼滤波&#xff09; 3、EKF&#xff08;可扩展卡尔曼滤波&#xff09; 4、UKF&#xff08;无迹卡尔曼滤波&#xff09; 5、PF&#xff08;粒子滤波&#xff09; 三、不同滤…

zabbix配置snmp trap--使用snmptrapd和Bash接收器--图文教程

1.前言 我的zabbix的版本是5.0版本&#xff0c;5.0的官方文档没有使用bash接收器的示例&#xff0c;6.0的官方文档有使用bash接收器的示例&#xff0c;但是&#xff0c;下载文件的链接失效&#xff1f;&#xff01; 这里讲解zabbix-server端配置和zabbix web端配置 2.zabbix-…

Day04 Liunx高级系统设计4-信号

进程间通讯 引入 如何将 A 进程中的数据传入 B 进程呢 ? 我们要使用进程间通讯 概述 中文名 : 进程间通讯 英文名 :IPC 英文全称 :Inter Processes Communication 作用: 数据传输&#xff1a;一个进程需要将他的数据发送给另一个进程】 资源共享&#xff1a;多个进程可以…

排序:直接选择排序

直接选择排序&#xff1a; 本质&#xff1a; 直接选择排序的本质就是在数组中进行遍历挑选出最大的元素&#xff0c;讲最大的元素放到对应的位置后&#xff0c;再次选出次大的位置&#xff0c;而后又放到对应的位置..........................直到数组成为一个有序序列。 优…

二叉树的层平均值[中等]

优质博文&#xff1a;IT-BLOG-CN 一、题目 给定一个非空二叉树的根节点 root , 以数组的形式返回每一层节点的平均值。与实际答案相差 10-5 以内的答案可以被接受。 示例 1&#xff1a; 输入&#xff1a;root [3,9,20,null,null,15,7] 输出&#xff1a;[3.00000,14.50000,1…

解决:AttributeError: module ‘copy’ has no attribute ‘copy’

解决&#xff1a;AttributeError: module ‘copy’ has no attribute ‘copy’ 文章目录 解决&#xff1a;AttributeError: module copy has no attribute copy背景报错问题报错翻译报错位置代码报错原因解决方法方法一方法二方法三今天的分享就到此结束了 背景 在使用之前的代…

C语言指针详解上

1 野指针 int main01(){//野指针就是没有初始化的指针,指针的指向是随机的,不可以 操作野指针//int a 0;//指针p保存的地址一定是定义过的(向系统申请过的)int *p;//野指针*p 200;printf("%d\n",*p);system("pause");return 0;}2 空指针 空指针的作用…

Unity 关于Ray、RaycastHit、Raycast及其使用

Unity中&#xff0c;我们要进行物理模拟和碰撞检测时&#xff0c;有三个重要的概念Ray、RaycastHit、Raycast。 其中&#xff0c;Ray可以理解为射线&#xff0c;它是一条从起点沿着特定方向延伸的无限长线段。 它的语法是&#xff1a; Ray(Vector3 origin, Vector3 directio…

使用阿里巴巴同步工具DataX实现Mysql与ElasticSearch(ES)数据同步

一、Linux环境要求 二、准备工作 2.1 Linux安装jdk 2.2 linux安装python 2.3 下载DataX&#xff1a; 三、DataX压缩包导入&#xff0c;解压缩 四、编写同步Job 五、执行Job 六、定时更新 6.1 创建定时任务 6.2 提交定时任务 6.3 查看定时任务 七、增量更新思路 一、Linux环境要…

el-table操作栏按钮过多 增加展开/收起功能

是的 如图所示有那么一条数据 列表操作栏的按钮七八个 小屏笔记本啥数据项也别看了 就剩下个固定列大刺刺的占着整个页面 解决方法&#xff1a; <el-table-column :width"tableToggle ? 600 : 300" label"操作栏" align"center" header-ali…

类和对象,this指针

一、类的引入&#xff1a; 如下&#xff0c;在C中&#xff0c;我们可以在结构体中定义函数&#xff0c;如下&#xff0c;之前我们学习C中中一直是在结构体中定义变量。 struct student{void studentinfo(const char* name,const char* gener,int age){ strcpy(_name,name);st…

前端css面试题(四)

文章目录 对一些 CSS 默认值的考察css选择器说一下z-indexz-index的值大的dom一定能覆盖z-index值小的dom吗如果一个第三方组件的z-index与我们现有业务的页面有冲突&#xff0c;怎么处理关于浮动元素样式引入权重问题链接引入&#xff08;Link&#xff09;和 import注入的区别…

理解输出电压纹波和噪声:来源与抑制

医疗设备、测试测量仪器等很多应用对电源的纹波和噪声极其敏感。理解输出电压纹波和噪声的产生机制以及测量技术是优化改进电路性能的基础。 1&#xff1a;输出电压纹波 以Buck电路为例&#xff0c;由于寄生参数的影响&#xff0c;实际Buck电路的输出电压并非是稳定干净的直流…

【wvp】测试记录

ffmpeg 这是个莫名其妙的报错&#xff0c;通过排查&#xff0c;应该是zlm哪个进程引起的 会议室的性能 网络IO也就20M

【分布式微服务专题】从单体到分布式(二、SpringCloud整合Nacos)

目录 前言阅读对象阅读导航前置知识笔记正文一、下载安装二、项目整合2.1 服务注册与发现2.2 动态配置管理 三、其他实验四、服务之间的调用 学习总结感谢 前言 本篇笔记主要是记录我整合Nacos项目进来的过程。以实现服务注册发现&#xff0c;以及分布式配置管理。关于Nacos&a…

Leetcode刷题笔记题解(C++):LCR 121. 寻找目标值 - 二维数组

思路&#xff1a;从左小角或者右上角开始遍历&#xff0c;假设右上角开始遍历&#xff0c;如果当前值大于目标值则列-1&#xff1b;如果当前值小于目标值则行1&#xff0c;以此遍历来查找目标值&#xff1b;注意col和row的选取 class Solution { public:bool findTargetIn2DPl…