C语言:字符函数 字符串函数 内存函数

C语言:字符函数 & 字符串函数 & 内存函数

    • 字符函数
      • 字符分类函数
      • 字符转换函数
        • tolower
        • toupper
    • 字符串函数
      • strlen
      • strcpy
      • strcat
      • strcmp
      • strstr
      • strtok
    • 内存函数
      • memcpy
      • memmove
      • memset
      • memcmp


字符函数

顾名思义,字符函数就是作用于字符的函数,而字符函数主要分为字符分类函数以及字符转换函数

字符分类函数

字符分类函数,用于判断一个字符是否属于某一个类

函数参数符合条件
isntrl任何字符
isspace空白字符,空格'',换页符'\f',换行符'\n',回车符'\r',制表符'\t',垂直制表符'\v'
isdigit十进制数字0 - 9
isxdigit十六进制数字,包括0 - 9a - fA - F
islower小写字母a - z
isupper大写字母A - Z
isalpha字母a - zA - Z
isalnum字母或数字,a - zA - Z0 - 9
ispunct标点符号,任何不属于字母或数字的可打印字符
isgraph任何图形字符
isprint任何可打印字符

对于以上函数,只要传入的字符符合要求,返回非0数字;否则返回0;


字符转换函数

tolower

将传入的大写字母变成小写字母

toupper

将传入的小写字母变成大写字母


字符串函数

strlen

strlen函数用于求出字符串的长度,其判定结束的标志为\0

大部分情况下,\0出现在字符串的结尾,但是如果\0出现在字符串的中间,那么我们就无法得到字符串的正确长度。


strcpy

在这里插入图片描述
strcpy函数用于拷贝字符串,其参数如下:

destination:拷贝后字符串存放的空间
source:字符串的来源

返回值为char*,指向拷贝后的destination,方便链式访问

  • 对于source来说,其结束的标志依然为'\0',其默认带有的'\0'也会被拷贝到指定字符串的末尾。

  • destination必须指向可以修改的空间,并且足以放置source的字符个数。

strncpy是一个与strcpy功能一致的函数,但是其可以限制拷贝字符的个数
对于strcpy,其拷贝字符的个数就是source'\0'之前字符个数。strncpy可以传入第三个参数,用于规定拷贝字符的个数。

示例:

char arr[100] = { 0 };
strncpy(arr, "hello", 3);

以上代码把"hello"的前三个字符”hel“拷贝进了arr中。

另外,如果字符串的长度不足n,那么少的位置会用\0补充


strcat

在这里插入图片描述
strcat函数用于对字符串进行追加,其会把source追加到destination的末尾。所以destinationsource都必须由\0结尾

示例:

char arr[100] = "hello";
strcat(arr, "world");

追加后,arr内部存储的就是helloworld了。

注意事项:

  • C语言标准没有要求此函数可以自追加,这取决于编译器的实现
  • destination必须是可以修改的

同样的,这个函数也有限制长度的版本strncat。第三个参数用于限定追加字符串的长度:当字符串长度超过n,则只追加n个字符;当字符串长度小于n,那么追加完这个字符串就不再追加


strcmp

在这里插入图片描述
此函数用于比较两个字符串的大小

  • str1大于str2,返回大于0的值
  • str1等于str2,返回0
  • str1小于str2,返回小于0的值

比较的是两个字符串的字典序,而非长度。

同样的,该函数存在一个限制长度的版本strncmp,比较两个字符串的前n个字符。


strstr

在这里插入图片描述
此函数用于进行字符串查找,即在str1中查找str2

如果找到了,返回str1中指向str2的指针
如果没找到,返回NULL


strtok

在这里插入图片描述
strtok函数用于对字符串进行分割,其有两个参数:

str:被分割的字符串
delimiters:分隔符组成的字符串

比如delimiters如果是“@#!”,那么这个字符串遇到#@!时就会将字符串进行分隔。

当在字符串中找到了对应的分隔符:

  1. 将这个分隔符改为\0
  2. 返回指向这个字符的指针

如果没找到字符,或者遇到末尾,此时返回NULL

另外地,strtok如果对str传入了NULL,下一次会从上一次更改的地方开始查找

示例:

char arr[100] = "hello!world?cs@dn";
const char* p = "!?@";for (char* r = strtok(arr, p); r != NULL; r = strtok(NULL, p))
{printf("%s\n", r);
}

以上代码中,第一次调用strtok传入了arr,后续都传入NULL,从而完成整个字符串的分隔:
输出结果:

hello
world
cs
dn

内存函数

C语言中的内存函数用于对内存进行操控,主要包括内存的拷贝,初始化,以及内存数据的比较。

memcpy

在这里插入图片描述

函数用于对内存进行拷贝,此函数包含三个参数:

destination:目的地,即拷贝后的数据存入的地方
source:源,即被拷贝的内存
num:字节数

示例:

int arr1[] = { 1,2,3,4,5 };
int arr2[5] = { 0 };memcpy(arr2, arr1, sizeof(int) * 5);

以上代码,完成了两个数组之间的内存拷贝,拷贝方向为:从arr1拷贝到arr2,共拷贝了sizeof(int) * 5 = 20个字节。

现在我们模拟实现一个memecpy

void* my_memcpy(void* destination, void* source, size_t num)
{void* ret = destination;assert(destination && source);while (num--){*(char*)destination = *(char*)source;destination = (char*)destination + 1;source = (char*)source + 1;}return ret;
}

可以看到,memcpy的拷贝是以字节为单位,将传入的void*指针转为char*指针,然后一个一个字节进行拷贝。这样这个函数就可以处理任何类型的内存拷贝。

但是这个函数存在一个问题,无法拷贝内存发生重叠的函数

看到以下过程:

int arr[6] = { 1,2,3,4 };memcpy(arr + 2, arr, sizeof(int) * 4 ;

此代码希望把arr的前四个元素拷贝到后四位去,即:

1 2 1 2 3 4

实际输出结果:

1 2 1 2 1 2

为什么会这样?

看到以下过程:
在这里插入图片描述
我们要把蓝色区域的数值拷贝到红色区域,从左往右拷贝,第一次进行拷贝,由于内存重叠,1 2会把3 4覆盖,导致后续拷贝3 4时拷贝到的还是1 2

C语言标准库没有要求memcpy对重叠的内存进行处理,但是有的编译器可以处理这种情况,有的不可以。
为处理这个情况,C语言有专门的函数memmove用于处理重叠内存的情况。


memmove

memmove函数也用于进行内存拷贝,与memcpy不同的是,C标准规定其可以处理内存的重叠。

示例:

int arr[6] = { 1,2,3,4 };memmove(arr + 2, arr, sizeof(int) * 4 ;

同样的代码,输出结果为:

1 2 1 2 3 4

这是如何做到的?
其实就是一个简单的拷贝方向的问题:

当我们把以上的重叠拷贝,从右向左拷贝:
在这里插入图片描述
可以看到,虽然发生了内存的重叠,但是我们先把会被覆盖的数据拷贝好,然后再覆盖重叠区域的数据,此时我们的拷贝就正常运行了。

所以memmove的实现中,我们要根据内存的重叠情况,来控制内存的拷贝方向,以保证重叠的数据先被拷贝。

代码如下:

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

destination指针小于source,我们就从低地址向高地址拷贝
destination指针大于source,我们就从高地址向低地址拷贝


memset

在这里插入图片描述
memset函数用于对内存进行初始化,其包含三个参数:

ptr:指向待初始化内存的指针
value:希望内存被初始化的值
num:想要初始化内存的字节数

此函数只能按照字节初始化内存,比如以下代码:

int arr[5] = { 0 };
memset(arr, 1, sizeof(int) * 5);

以上代码并不是把数组中的五个int元素初始化为1,而是把每个字节都初始化为00000001


memcmp

在这里插入图片描述
此函数用于比较内存中的数据,ptr1ptr2是被比较的内存。
其按照字节进行比较,从前往后,一个一个字节比较ASCII码值,当某一对字节的ASCII码值不同,此时该字节ASCII码值大的内存,就是比较大的。


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

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

相关文章

1.1 编程环境的安装

汇编语言 汇编语言环境部署 第二个运行程序直接双击安装一直下一步即可MASM文件复制到D盘路径下找到dosbox安装路径&#xff1a;C:\Program Files (x86)\DOSBox-0.74找到该文件双击打开它&#xff0c;修改一下窗口大小 把这两行改成如下所示 运行dos&#xff0c;黑框中输入mou…

打造去中心化透明储蓄罐:Solidity智能合约的又一实践

一、案例背景 传统的储蓄罐通常是由个人或家庭使用&#xff0c;用于存放硬币或小额纸币。然而&#xff0c;这样的储蓄罐缺乏透明性&#xff0c;用户无法实时了解储蓄情况&#xff0c;也无法确保资金的安全性。 通过Solidity智能合约&#xff0c;我们可以构建一个去中心化…

园区能耗监测管控平台

园区能耗监测管控平台是一种基于先进科技的能源管理系统&#xff0c;旨在帮助园区实现能源消耗的精准监测和高效管控。这一平台集成了能耗监测、数据分析、远程控制等功能&#xff0c;为园区管理者提供了全方位的能源管理解决方案&#xff0c;助力园区实现节能减排、降低成本的…

vue3使用echarts绘制地图

vue3使用echarts绘制地图 安装echarts npm install echarts下载地图的json数据【我这里是把json数据单独粘出来然后新建了一个文件china.json】 下载中国及各个省份的地图数据引入 import chinaJson from ./china.json绘制地图 <template><div ref"myChart&q…

Windows WMI详解

WMI简介 WMI ( Windows Management Instrumentation, Windows管理规范)是Windows 2000/XP管理系统的核心&#xff0c;属于管理数据和操作的基础模块。设计WMI的初衷是达到一种通用性&#xff0c;通过WM操作系统、应用程序等来管理本地或者远程资源。它支持分布式组件对象模型(…

sora技术报告阅读

sora是一个在可变持续时间、分辨率和宽高比的视频和图像上联合训练文本条件扩散模型。 需要将所有类型的视觉数据转化为统一表示的方法&#xff0c;使得能够对生成模型进行大规模训练。 Sora是一个通用的视觉数据模型&#xff0c;它可以生成不同持续时间、宽高比和分辨率的视…

打造透明银行存储:Solidity智能合约的实践与探索

引言&#xff1a; 随着区块链技术的快速发展&#xff0c;智能合约作为其中的核心组件&#xff0c;正被越来越多地应用于各种场景。作为智能合约的编程语言&#xff0c;Solidity因其对以太坊平台的深度支持而备受关注。在这篇文章中&#xff0c;我们将通过构建一个透明的银行存储…

webrtc

stun服务 阿里云服务器安全组添加端口开放 webrtc-streamer视屏流服务器搭建 - 简书

docker安装单机版canal和使用

说明&#xff1a;我安装的组件架构如下&#xff1a; 1、准备一台虚拟机&#xff0c;192.168.2.223&#xff0c;我安装的时候&#xff0c;docker只支持canal1.1.6版本&#xff0c;1.1.7无法使用docker安装.还有一点要补充&#xff0c;就是1.1.6好像不支持es8.0以上版本&#x…

Java+SpringBoot,打造社区疫情信息新生态

✍✍计算机编程指导师 ⭐⭐个人介绍&#xff1a;自己非常喜欢研究技术问题&#xff01;专业做Java、Python、微信小程序、安卓、大数据、爬虫、Golang、大屏等实战项目。 ⛽⛽实战项目&#xff1a;有源码或者技术上的问题欢迎在评论区一起讨论交流&#xff01; ⚡⚡ Java实战 |…

FreeRTOS 信号量

目录 一、信号量的概念 二、二值信号量 1、二值信号量的定义 2、二值信号量的作用 3、二值信号量的操作 4、二值信号量操作实验 5、二值信号量会导致优先级翻转问题 三、互斥信号量 1、互斥信号量的引入 2、注意&#xff1a;互斥信号量不能用于中断服务函数中&#xf…

【STM32】STM32学习笔记-独立看门狗和窗口看门狗(47)

00. 目录 文章目录 00. 目录01. WDG概述02. 独立看门狗相关API2.1 IWDG_WriteAccessCmd2.2 IWDG_SetPrescaler2.3 IWDG_SetReload2.4 IWDG_ReloadCounter2.5 IWDG_Enable2.6 IWDG_GetFlagStatus2.7 RCC_GetFlagStatus 03. 独立看门狗接线图04. 独立看门狗程序示例105. 独立看门…