【再识C进阶3(下)】详细地认识字符分类函数,字符转换函数和内存函数

前言

💓作者简介: 加油,旭杏,目前大二,正在学习C++数据结构等👀
💓作者主页:加油,旭杏的主页👀

⏩本文收录在:再识C进阶的专栏👀

🚚代码仓库:旭日东升 1👀

🌹欢迎大家点赞 👍 收藏 ⭐ 加关注哦!💖

学习目标:

       在上一篇博客中,我们学习了字符串函数,字符串是由字符构成的,那么这篇博客将会为大家讲解字符分类函数字符转换函数以及内存函数,那么字符就又有了一片新天地。这篇博客还是会很简单的,废话不多说,我们来开始写博客。


学习内容:

通过上面的学习目标,我们可以列出要学习的内容:

  1. 字符分类函数
  2. 字符转换函数
  3. 内存函数 

一、字符分类函数

先来看一下全部字符分类函数的大纲,总共有12个,请看下面表格:

函数如果他的参数符合下列条件就返回真
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任何可打印的字符,包括圆形字符和空白字符

1.1 iscntrl字符函数

1.1.1 控制字符的概念(了解一下)

       在小编刚听到这个函数时,我在纳闷控制字符是什么?控制字符的概念简单来讲,就是出现于特定的信息文本中,表示某一控制功能的字符。

       精确来讲,就是在ASCII码中,第0~31号及第127号(共33个)是控制字符或通讯专用字符,如控制符:LF(换行)、CR(回车)、FF(换页)、DEL(删除)、BS(退格)、BEL(振铃)等;通讯专用字符:SOH(文头)、EOT(文尾)、ACK(确认)等。

1.1.2 iscntrl函数的作用

       其功能就是检查字符c是否为控制字符,如果是,则返回真(非0的数,有可能是负数),如果不是,则返回假。 

1.2  isspace字符函数

       其功能是检查字符c是否为空白字符,空白字符的概念在上面表格已经写的很详细了,当然在图中也很清楚。 

1.3  isdigit字符函数

       这个函数还是用的比较多的,因为其功能是检查字符c是否为十进制数字,十进制数字想必大家都很清楚,所以这个函数还算能用的上(大家还是要重视一下)。

1.4 isxdigit字符函数

       这个函数的范围和上面的isdight字符函数差不多,所以小编推断这个函数应该会用的很少,但还是要看一下,有个印象。这个函数的功能是检查字符c是否为十六进制数字,包括所有的十进制数字,小写字母 a~f,大写字母 A~F。

1.5  islower字符函数

       这个函数就有点意思,功能是检查字符c是否为小写字母,在一些题目中可能会用到这个函数,这个函数还是要记住为好。

举个例子:

题目:请回答出你所你输入的字符串中小写字母的个数为多少?

代码:

#include <stdio.h>
#include <ctype.h>
int main()
{int n = 0;scanf("%d", &n);char arr[20];for (int i = 0; i < n; i++){scanf("%c", &arr[i]);}int count = 0;for (int i = 0; i < n; i++){if (islower(arr[i])){count++;}}printf("%d\n", count);return 0;
}

1.6 isupper字符函数

       有了检查是否为小写字母的函数,那必然会有检查大写字母的字符函数,同样,我们在网站来看看这个函数吧!这个用法是和上面的字符函数的用法是基本一样的,这里就不在赘述。

1.7  isalpha字符函数

       这个函数就是将 islower 字符函数和 isupper 字符函数结合起来,其功能是将检查是否为26个英文字母。个人感觉不是很好用,因为其的范围实在是太大了。

二、字符转换函数

       字符转换函数一共有两个,一个是 tolower 字符函数,一个是 toupper 字符函数。这两个字符还是比较重要的,因为在题目的求解中会用到这两个字符函数。我们需要先来了解这两个字符函数的功能之后,我们再来用一个习题进行巩固练习。

2.1 tolower字符函数

2.1.1 函数功能

       这个函数将判断所选的字符c是否为大写字母,如果是大写字母的话,将这个字符c转换为小写字母;如果不是大写字母,将不会转换这个字符c。大致工作原理就是利用 islower 字符函数进行判断,然后将这个字符函数进行处理即可。

2.1.2 函数例题

题目:

       将一个字符串中的所有大写字母变成对应的小写字母,其它字母不变。例如:对字符串“ABC12! EF”执行函数后输出结果为“abc12! ef”。

代码:

       这道题目的代码, 我们用两种方式进行解决,一种是运用库函数tolower,一种是不运用这个函数,我们来进行一些区别。

//如果使用库函数, 写的代码
void toUp(char* str)
{int len = strlen(str);for (int i = 0; i < len; i++){if (isupper(*(str + i))){*(str + i) = tolower(*(str + i));}}
}
//如果不用库函数,代码会是什么样子的
void toUp(char* str)
{int len = strlen(str);for (int i = 0; i < len; i++){if (str[i] >= 'a' && str[i] <= 'z'){str[i] = str[i] - 32;}}
}

2.2 isupper字符函数

2.2.1 函数功能

       这个函数将判断所选的字符c是否为小写字母,如果是小写字母的话,将这个字符c转换为大写字母;如果不是小写字母,将不会转换这个字符c。大致工作原理就是利用 isupper 字符函数进行判断,然后将这个字符函数进行处理即可。

2.2.2 函数例题

题目:

       将一个字符串中的所有小写字母变成对应的大写字母,其它字母不变。例如:对字符串“abc12! ef”执行函数后输出结果为“ABC12! EF”。

代码:

       这道题目的代码, 我们用两种方式进行解决,一种是运用库函数tolower,一种是不运用这个函数,我们来进行一些区别。

//如果使用库函数, 写的代码
void toUp(char* str)
{int len = strlen(str);for (int i = 0; i < len; i++){if (islower(*(str + i))){*(str + i) = toupper(*(str + i));}}
}
//如果不用库函数,代码会是什么样子的
void toUp(char* str)
{int len = strlen(str);for (int i = 0; i < len; i++){if (str[i] >= 'a' && str[i] <= 'z'){str[i] = str[i] - 32;}}
}

2.3 最后进行总结一下

       在这两个函数中,我们可以看见这两个函数的返回类型和形式参数基本是一样的,但是我们返回的是字符,而不是整形数字,为什么这两个函数的返回类型是 int 呢?

三、内存函数

       内存函数就是内存相关的函数,这些函数有4个:memcpymemmovememsetmemcmp

       如果我们将这种函数和上一篇字符串函数是有点相似的,但是我们已经有了字符串函数进行拷贝了,为什么还要用内存函数进行拷贝呢?大家要把视野放宽一下,在内存中不仅仅只有字符串,还有其他类型的数据需要拷贝!那么我们开始进行学习吧!

3.1 memcpy内存函数

3.1.1 函数功能

       这个函数的功能是复制内存块,将从源头指向的位置开始往后的num个字节复制到目标指向的内存块中。乍一看,感觉和strcpy字符函数的功能有点像,但是他们两个还是有不同,请看下面的对比图:

3.1.2 函数使用 

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
int main()
{int arr1[5];int arr2[] = { 0,1,2,3,4 };memcpy(arr1, arr2, 5 * sizeof(int));for (int i = 0; i < 5; i++){printf("%d ", arr1[i]);}return 0;
}

3.1.3 模拟实现一下memcpy函数

void my_memcpy(void* dest, const void* src, size_t num)
{assert(dest && src);while (num--){*(char*)dest = *(char*)src;dest = (char*)dest + 1;src = (char*)src + 1;}
}

       当看完模拟实现的过程后,我们可以看出在while循环中指针的变化可能不止一种写法,但是其他的写法会在不同的编译器中会出现漏洞。比如说下面的代码在c文件中可以编译过去,而在cpp文件中编译不过去。

void* my_memcpy(void* dest, const void* src, size_t num)
{assert(dest && src);void* ret = dest;while (num--){*(char*)dest = *(char*)src;((char*)dest)++;((char*)src)++;}return dest;
}

3.1.4 有关这个函数的一些问题

       在这个函数的开头,小编放了一张图片,在其上面我们有一些文字。小编再来解释一下什么意思?在前面我们学习了qsort函数,是不是与这个函数的思想有点相似呢?

  • 为什么他的类型是 void* 呢?因为在C语言中,这个函数将会拷贝所有内存块中的数据,所以他的类型必须是void*,这样其就能返回任何类型的数据。
  • 为什么 num 的单位是字节呢?因为数据类型中最小的单位是一个字节,如果我们一个字节一个字节的进行拷贝,我们就可以将所有类型的数据进行拷贝,这样的格局就大了。

3.2 memmove内存函数

       在前面我们学习了memcpy函数,发现其可以将内存中的数据拷贝到另一个内存中,那么我们现在有一个问题就是:加入一个数组arr,其内容放有1,2,3,4,5,6,7,7,9,10。   

       如果我们想要将1,2,3,4,5向后移动2格,可不可以用我们自己定义的memcpy函数呢?答案是否定的。我们可以通过调试进行验证。(为什么不用库函数中的memcpy函数,先卖个关子,之后会说)

       所以这就引出了这一部分我们要学习的函数——memmove函数,这个函数会将内存块中的数据进行移动,要与memcpy函数区分!接下来,我们来验证一下:

总结:

     memcpy函数是用于两个不重叠的内存中,将一个内存中的数据拷贝到另一个内存当中;memmove函数是用于一个重叠的内存中,将这个内存中的一些数据移动位置。

3.2.1 memcpy函数居然也能完成memmove函数的功能?

       为什么小编会这样说呢?因为在前言中,小编用的是自己模拟实现的memcpy函数,而用的不是库函数的memcpy函数。如果我们用的是memcpy函数呢?结果是可以实现memmove函数的功能。难道是我们写错了吗,不是,而是VS中的库函数的功能不叫强大(太卷了),现在我们来看一下memcpy函数实现功能:

3.2.2 函数功能

       这个函数的功能是从source指针指向的位置开始,拷贝num个字节的内存块到destination中,其是能够对本身进行覆盖拷贝的函数,其又同时兼备了 memcpy函数可做的事。 

3.2.3 模拟实现一下memmove内存函数

void* my_memmove(void* dest, const void* src, size_t num)
{assert(dest && src)void* ret = dest;if ((char*)dest < (char*)src){while (num--){*(char*)dest = *(char*)src;dest = (char*)dest + 1;src = (char*)src + 1;}}else{while (num--){*((char*)dest + num) = *((char*)src + num);}}return ret;
}

3.3 memset函数

       memset函数是用来设置内存的,以字节为单位进行设置内存(更适合字符数组)。但要注意这个memset函数的使用场景,因为场景不同,这个memset函数所实现的功能是不同的。

第一个场景:

       如果有一个数组arr,其内容是全为0,如果我们想让其的内容变为1,我们应该怎么办呢?我们如果使用memset函数,其结果又是什么呢?

       为什么会是这个样子呢?因为memset函数是一个字节一个字节地进行设置,会将每一个字节的内容都设置为1,所以四个字节的数字的值不会是1,。

第二个场景:

       如果我们想使数组的内容全初始化为0,我们可不可以使用memset函数呢?答案是可以的,因为即使每一个字节的内容都为0,那么四个字节的内容还是会等于0的。

3.4 memcmp内存函数

       这个函数的功能是比较两个内存块对应字节内容的大小,不过,这个函数一个字节一个字节地比较,所以会有一些不同常理的事情发生:


学习产出:

  1. 字符分类函数
  2. 字符转换函数
  3. 内存函数 

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

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

相关文章

vuereact质检工具(eslint)安装使用总结

1、ESLint ESLint工具主要类似java中的checkStyle和findbugs&#xff0c;是检查代码样式和逻辑规范的工具。 1.1、ESLint安装流程 打开VSCode软件&#xff0c;打开扩展中心&#xff0c;下载ESLint插件 图1.1 点击后面的install按进行安装&#xff0c;如图1.2所示&#xff1…

黑马VUE3视频笔记

目录 一、使用create-vue创建项目 二、setup选项 三、reactive和ref函数 1.reactive() 2.ref() 三、computed 四、watch ​五、生命周期函数 六、父传子、子传父 父传子defineProps 子传父defineEmits 七、模板引用 ref defineExpose 八、跨层传递普通数据 prov…

SpringBoot 学习(二)配置

2. SpringBoot 配置 2.1 配置文件类型 配置文件用于修改 SpringBoot 的默认配置。 2.1.1 properties 文件 **properties ** 是属性文件后缀。 文件名&#xff1a;application.properties 只能保存键值对。 基础语法&#xff1a;keyvalue namewhy注入配置类 Component //…

百度SEO优化不稳定的原因分析(提升网站排名的稳定性)

百度SEO优化不稳定介绍蘑菇号-www.mooogu.cn SEO不稳定是指网站在搜索引擎中的排名不稳定&#xff0c;随着时间的推移会发生变化。这种情况可能会出现在网站页面结构、内容质量、外链质量等方面存在缺陷或不合理之处。因此&#xff0c;优化SEO非常重要&#xff0c;可以提高网站…

Cruise 的界面和模型文件路径介绍

文章目录 打开 Cruise自带模型所在路径自带模型分类Cruise 中的模型路径解析打开用户手册建模界面介绍打开模型 打开 Cruise 最新的几个 Cruise 软件都是集成到一个平台上的&#xff0c;名为 AVL Advanced Simulation Desktop。 自带模型所在路径 User 选项卡下的模型&#x…

Spring实现简单的Bean容器

1.BeanDefinition&#xff0c;用于定义 Bean 实例化信息&#xff0c;现在的实现是以一个 Object 存放对象 public class BeanDefinition {/*** bean对象*/private Object bean;/*** 存放 &#xff08;定义&#xff09;Bean 对象*/public BeanDefinition(Object bean) {this.bea…

关于TUM数据集

2、验证回环检测算法&#xff0c;需要有人工标记回环的数据集。然而人工标记回环是很不方便的&#xff0c;我们会考虑根据标准轨迹计算回环。即&#xff0c;如果轨迹中有两个帧的位姿非常相近&#xff0c;就认为它们是回环。请根据TUM数据集给出的标准轨迹&#xff0c;计算出一…

怎么选择伪原创工具?伪原创工具推荐

什么是伪原创工具&#xff1f;伪原创工具是一种可以将已有文本进行修改、改写或重新组合&#xff0c;生成新的文本内容的工具。 伪原创工具的作用 节省时间和精力&#xff1a;手工创作内容需要耗费大量时间和精力&#xff0c;而伪原创工具可以在短时间内生成大量内容&#xf…

嵌入式中如何用C语言操作sqlite3(07)

sqlite3编程接口非常多&#xff0c;对于初学者来说&#xff0c;我们暂时只需要掌握常用的几个函数&#xff0c;其他函数自然就知道如何使用了。 数据库 本篇假设数据库为my.db,有数据表student。 nonamescore4嵌入式开发爱好者89.0 创建表格语句如下&#xff1a; CREATE T…

C++中实现雪花算法来在秒级以及毫秒及时间内生成唯一id

1、雪花算法原理 雪花算法&#xff08;Snowflake Algorithm&#xff09;是一种用于生成唯一ID的算法&#xff0c;通常用于分布式系统中&#xff0c;以确保生成的ID在整个分布式系统中具有唯一性。它的名称来源于雪花的形状&#xff0c;因为生成的ID通常是64位的整数&#xff0…

【数据结构】哈希表(详)

文章目录 前言正文一、基本概念二、基本原理1.哈希函数1.1直接定址法&#xff08;常用&#xff09;1.2除留余数法&#xff08;常用&#xff09;1.3 平方取中法&#xff08;了解&#xff09;1.4 折叠法(了解)1.5 随机数法(了解)1.6数学分析法(了解) 2.哈希冲突2.1 平均查找长度2…

QT配置MySQL数据库 ninja: build stopped: subcommand failed

QT配置MySQL数据库 我当前的软件版本&#xff1a;QT Creator 10.0.2 (community)&#xff0c;MingW 6.4.3 (QT6)&#xff0c;MySQL 8.0。 MySQL不配置支持的数据库有QList("QSQLITE", "QODBC", "QPSQL")&#xff0c;这个时候是不支持MYSQL数据…

No127.精选前端面试题,享受每天的挑战和学习

🤍 前端开发工程师(主业)、技术博主(副业)、已过CET6 🍨 阿珊和她的猫_CSDN个人主页 🕠 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 🍚 蓝桥云课签约作者、已在蓝桥云课上架的前后端实战课程《Vue.js 和 Egg.js 开发企业级健康管理项目》、《带你从入…

箱讯科技成功闯入第八届“创客中国”全国总决赛—在国际物流领域一枝独秀

添加图片注释&#xff0c;不超过 140 字&#xff08;可选&#xff09; 2023年9月26日&#xff0c;第八届“创客中国”数字化转型中小企业创新创业大赛决赛在贵州圆满收官。 经过初赛、复赛、决赛的激烈角逐&#xff0c;箱讯科技与众多强劲对手同台竞技&#xff0c;最终凭借出…

Android gradle dependency tree change(依赖树变化)监控实现

文章目录 前言基本原理执行流程diff 报告不同分支 merge 过来的 diff 报告同个分支产生的 merge 报告同个分支提交的 diff 报告 具体实现原理我们需要监控怎样的 Dendenpency 变化怎样获取 dependency Treeproject.configurations 方式./gradlew dependenciesAsciiDependencyRe…

铁路用热轧钢轨

声明 本文是学习GB-T 2585-2021 铁路用热轧钢轨. 而整理的学习笔记,分享出来希望更多人受益,如果存在侵权请及时联系我们 1 范围 本标准规定了铁路用钢轨的订货内容、分类、尺寸、外形、质量及允许偏差、技术要求、试验方法、检 验规则、标志及质量证明书。 本标准适用于3…

iMovie for Mac v10.3.9(视频剪辑)

iMovie是一款视频剪辑软件&#xff0c;广泛应用于Mac和iOS设备。以下是关于iMovie软件的一些推荐信息&#xff1a; 简单易用。iMovie的设计简洁&#xff0c;操作简单&#xff0c;即使是没有剪辑经验的新手也可以轻松上手。软件内置了丰富的视觉效果、滤镜、绿幕抠图、分屏和画…

【腾讯云国际站】CDN内容分发网络特性介绍

为什么使用腾讯云国际站 CDN 内容分发网络&#xff1f; 当用户直接访问源站中的静态内容时&#xff0c;可能面临的体验问题&#xff1a; 客户离服务器越远&#xff0c;访问速度越慢。客户数量越多&#xff0c;网络带宽费用越高。跨境用户访问体验较差。 腾讯云国际站CDN 如何改…

Ctfshow web入门 XSS篇 web316-web333 详细题解 全

CTFshow XSS web316 是反射型 XSS 法一&#xff1a; 利用现成平台 法二&#xff1a; 自己搭服务器 先在服务器上面放一个接受Cookie的文件。 文件内容&#xff1a; <?php$cookie $_GET[cookie];$time date(Y-m-d h:i:s, time());$log fopen("cookie.txt"…

MySQL学习笔记19

MySQL日志文件&#xff1a;MySQL中我们需要了解哪些日志&#xff1f; 常见日志文件&#xff1a; 我们需要掌握错误日志、二进制日志、中继日志、慢查询日志。 错误日志&#xff1a; 作用&#xff1a;存放数据库的启动、停止和运行时的错误信息。 场景&#xff1a;用于数据库的…