高阶C语言|深入理解字符串函数和内存函数
文章目录
- 前言
- 1.求字符串长度
- 1.1 字符串长度函数:`strlen`
- 模拟实现
- 长度不受限制的字符串函数
- 1.2 字符串拷贝函数:`strcpy`
- 模拟实现
- 1.3 字符串连接函数:`strcat`
- 模拟实现
- 1.4 字符串比较函数:`strcmp`
- 模拟实现
- 长度受限制的字符串函数
- 2.1`strncpy`
- 2.2`strncat`
- 2.3`strncmp`
- 字符串查找
- 3.1字符串查找函数:`strstr`
- 模拟实现
- 3.2字符串分割函数:`strtok`
- 错误信息报告
- 错误信息报告函数:`strerror`
- 内存操作函数
- 4.1 内存操作函数:`memcpy` 和 `memmove`
- `memcpy`模拟实现
- `memmove`模拟实现
- 4.2 内存操作函数:`memset`
- 注意事项:
- 4.3 `memset` 的应用场景
- 4.4 内存比较函数:`memcmp`
- 使用示例:
- 输出:
- 注意事项:
- 应用场景:
前言
在C语言中,字符和字符串是常用的数据类型。然而,C语言并没有专门的字符串类型,所有字符串都是通过字符数组或字符串常量来表示。为了处理这些字符串,C语言提供了许多强大的库函数。本文将详细介绍这些常用的字符和字符串处理函数,以及它们的使用方法和注意事项。
1.求字符串长度
1.1 字符串长度函数:strlen
strlen 函数用来计算字符串的长度。它返回字符串中不包含终止字符 '\0' 的字符数量。
size_t strlen(const char *str);
使用示例:
#include <stdio.h>
#include <string.h>int main() {const char *str = "Hello, world!";printf("Length of the string: %zu\n", strlen(str)); // 输出:13return 0;
}
模拟实现
#include<stdio.h>
#include<assert.h>
//#include<string.h>int my_strlen(const char* str)
{assert(str);int count = 0;while (*str++) {count++;}return count;
}
int main()
{char arr[] = "abcdef";int ret = my_strlen(arr);printf("%d", ret);return 0;
}
长度不受限制的字符串函数
1.2 字符串拷贝函数:strcpy
strcpy 函数将源字符串拷贝到目标字符串,包括终止符 '\0'。
char *strcpy(char *destination, const char *source);
注意事项:
- 源字符串必须以
'\0'结束。 - 目标空间必须足够大,以容纳源字符串。
使用示例:
#include <stdio.h>
#include <string.h>int main() {char dest[20];strcpy(dest, "Hello");printf("Destination string: %s\n", dest); // 输出:Helloreturn 0;
}
模拟实现
#include<stdio.h>
//#include<string.h>
#include<assert.h>void my_strcpy(char* dest,const char* source)
{assert(dest && source);while (*dest++=*source++) {;}
}int main()
{char arr1[] = "abcdefg";char arr2[] = "hijklmn";my_strcpy(arr1, arr2);printf("%s", arr1);return 0;
}
1.3 字符串连接函数:strcat
strcat 函数将源字符串追加到目标字符串的末尾。它会覆盖目标字符串末尾的 '\0',并将新的字符串结束符 '\0' 放置在新字符串的末尾。
char *strcat(char *destination, const char *source);
使用示例:
#include <stdio.h>
#include <string.h>int main() {char dest[20] = "Hello, ";strcat(dest, "world!");printf("Concatenated string: %s\n", dest); // 输出:Hello, world!return 0;
}
模拟实现
#include<stdio.h>
//#include<string.h>
#include<assert.h>char* my_strcat(char* dest, const char* source)
{char* ret = dest;while (*dest) {dest++;}while (*dest++ = *source++) {;}return ret;
}int main()
{char arr1[20] = "abcdf";char arr2[10] = "ghijk";my_strcat(arr1, arr2);printf("%s", arr1);return 0;
}
1.4 字符串比较函数:strcmp
strcmp 函数比较两个字符串的大小。它返回一个整数,根据两个字符串的字典顺序进行比较。
int strcmp(const char *str1, const char *str2);
- 如果
str1大于str2,返回大于 0 的值。 - 如果
str1等于str2,返回 0。 - 如果
str1小于str2,返回小于 0 的值。
使用示例:
#include <stdio.h>
#include <string.h>int main() {const char *str1 = "abc";const char *str2 = "abd";int result = strcmp(str1, str2);if (result < 0) {printf("str1 is less than str2\n");} else if (result > 0) {printf("str1 is greater than str2\n");} else {printf("str1 is equal to str2\n");}return 0;
}
模拟实现
#include<stdio.h>
//#include<string.h>
#include<assert.h>int my_strcmp(const char* str1,const char* str2)
{assert(str1 && str2);while (*str1== *str2) {str1++;str2++;}return *str1 - *str2;
}int main()
{char arr1[] = "abcd";char arr2[] = "abcf";int ret = my_strcmp(arr1, arr2);if (ret == 0){printf("=\n");}else if (ret > 0) {printf(">\n");}elseprintf("<\n");return 0;
}
长度受限制的字符串函数
2.1strncpy
char * strncpy ( char * destination, const char * source, size_t num );
- 拷贝num个字符从源字符串到目标空间。
- 如果源字符串的长度小于num,则拷贝完源字符串之后,在目标的后边追加0,直到num个。
2.2strncat
char * strncat ( char * destination, const char * source, size_t num );
在目标字符串结尾追加源字符串前num个字符
2.3strncmp
int strncmp ( const char * str1, const char * str2, size_t num );
比较到出现另个字符不一样或者一个字符串结束或者num个字符全部比较完。
字符串查找
3.1字符串查找函数:strstr
strstr 函数查找一个字符串在另一个字符串中首次出现的位置。如果找到,则返回该位置的指针;如果未找到,则返回 NULL。
char *strstr(const char *haystack, const char *needle);
使用示例:
#include <stdio.h>
#include <string.h>int main() {const char *str = "This is a simple string";const char *substr = "simple";char *result = strstr(str, substr);if (result != NULL) {printf("Found substring: %s\n", result); // 输出:simple string}return 0;
}
模拟实现
#include<stdio.h>
//#include<string.h>
#include<assert.h>char* my_strstr(const char* str1, const char* str2)
{assert(str1 && str2);if (*str2 == '\0') {return (char*)str1;}const char* s1 = NULL;const char* s2 = NULL;const char* cp = str1;while (cp) {s1 = cp;s2 = str2;while (*s1 !='\0' && *s2 != '\0' && * s1 == *s2) {s1++;s2++;}if (*s2 == '\0')return (char*)cp;cp++;}return NULL;
}int main()
{char* p1 = "abbbcdef";char* p2 = "ba";char* ret = my_strstr(p1, p2);if (ret = NULL) {printf("Ҳ\n");}else {printf("%s", ret);}return 0;
}
3.2字符串分割函数:strtok
strtok 函数将字符串分割成多个标记,每次调用 strtok 返回下一个标记。当没有更多标记时,返回 NULL。
char *strtok(char *str, const char *delim);
注意事项:
strtok会修改原始字符串,因此一般会拷贝字符串后再进行分割。- 每次调用
strtok返回字符串中的下一个标记,后续调用需要将第一个参数设为NULL。
使用示例:
#include <stdio.h>
#include <string.h>int main() {char str[] = "Hello, world! C programming";char *token = strtok(str, " ,!");while (token != NULL) {printf("Token: %s\n", token);token = strtok(NULL, " ,!");}return 0;
}
错误信息报告
错误信息报告函数:strerror
strerror 函数根据错误码返回相应的错误信息。
char *strerror(int errnum);
使用示例:
#include <stdio.h>
#include <string.h>
#include <errno.h>int main() {FILE *file = fopen("nonexistent.txt", "r");if (file == NULL) {printf("Error: %s\n", strerror(errno));}return 0;
}
内存操作函数
4.1 内存操作函数:memcpy 和 memmove
memcpy用于从源内存区域复制指定字节的数据到目标内存区域。memmove与memcpy类似,但它处理内存重叠的情况。
void *memcpy(void *dest, const void *src, size_t n);
void *memmove(void *dest, const void *src, size_t n);
使用示例:
#include <stdio.h>
#include <string.h>int main() {char src[] = "This is a test.";char dest[20];memcpy(dest, src, strlen(src) + 1);printf("Copied string: %s\n", dest);return 0;
}
memcpy模拟实现
#include<stdio.h>
#include<string.h>
#include<assert.h>void* my_memcpy(void* dest, const void* source, size_t num)
{assert(dest && source);void* ret = dest;while (num--){*(char*)dest = *(char*)source;dest = (char*)dest + 1;source = (char*)source + 1;}return ret;
}void test1()
{int arr1[10] = { 1,2,3,4,5,6,7,8,9,10 };int arr2[10] = { 0 };my_memcpy(arr1, arr2, 20);
}int main()
{test1();return 0;
}
memmove模拟实现
#include<stdio.h>
#include<string.h>
#include<assert.h>void* my_memmove(void* dest, const void* source, size_t num)
{assert(dest && source);void* ret = dest;if (dest < source) {while (num--) {*(char*)dest = *(char*)source;dest = (char*)dest + 1;source = (char*)source + 1;}}else {while (num--) {*((char*)dest + num) = *((char*)source + num);}}return ret;
}void test1()
{int arr[10] = {1,2,3,4,5,6,7,8,9,10};my_memmove(arr+3, arr, 20);
}int main()
{test1();//test2();return 0;
}
4.2 内存操作函数:memset
memset 函数用于将一块内存区域的所有字节设置为指定的值。通常用于初始化内存或将内存区域清零。
void *memset(void *s, int c, size_t n);
s:指向内存区域的指针。c:要设置的值(以无符号字符形式)。n:要设置的字节数。
注意事项:
memset函数将指定的字节值填充到内存区域中,可以用于初始化数组或结构体的值。- 由于是按字节处理,因此它适用于任何类型的内存块。
使用示例:
#include <stdio.h>
#include <string.h>int main() {char str[20];memset(str, 'A', sizeof(str) - 1); // 将前19个字节设置为'A'str[19] = '\0'; // 确保字符串以'\0'结束printf("The string is: %s\n", str); // 输出:AAAAAAAAAAAAAAAAAAAreturn 0;
}
在上面的例子中,memset 将数组 str 的前 19 个字节设置为字符 'A',并在最后设置为字符串的终止符 '\0',确保它成为一个有效的字符串。
4.3 memset 的应用场景
-
初始化数组或结构体:在动态分配内存或创建数据结构时,使用
memset可以方便地初始化内存,将内存区域填充为特定的值。例如,常见的做法是使用memset来清零结构体或数组中的内容。 -
清空敏感数据:当处理涉及敏感信息(如密码、密钥)的程序时,使用
memset可以确保这些数据在使用后被立即清除,以减少安全隐患。 -
内存清零:在程序需要清除缓存或重置数据时,
memset可以帮助快速设置内存区域为零。
4.4 内存比较函数:memcmp
memcmp 函数用于比较两个内存区域的内容。它按字节逐一比较两个内存块,并根据比较结果返回一个整数值。
int memcmp(const void *ptr1, const void *ptr2, size_t num);
ptr1:指向第一个内存区域的指针。ptr2:指向第二个内存区域的指针。num:要比较的字节数。
返回值:
- 如果两个内存区域的内容完全相同,则返回
0。 - 如果
ptr1指向的内存块大于ptr2,则返回大于 0 的值。 - 如果
ptr1指向的内存块小于ptr2,则返回小于 0 的值。
使用示例:
#include <stdio.h>
#include <string.h>int main() {char buffer1[] = "DWgaOtP12df0";char buffer2[] = "DWGAOTP12DF0";int result = memcmp(buffer1, buffer2, sizeof(buffer1));if (result > 0) {printf("'%s' is greater than '%s'.\n", buffer1, buffer2);} else if (result < 0) {printf("'%s' is less than '%s'.\n", buffer1, buffer2);} else {printf("'%s' is the same as '%s'.\n", buffer1, buffer2);}return 0;
}
输出:
'DWgaOtP12df0' is greater than 'DWGAOTP12DF0'.
在这个示例中,memcmp 比较了两个字符数组 buffer1 和 buffer2 的内容。由于在比较过程中,DWgaOtP12df0 的字节值大于 DWGAOTP12DF0,所以返回的值大于 0。
注意事项:
- 按字节比较:
memcmp是按字节逐一比较内存区域的内容,因此它不会考虑数据的类型。如果要比较更复杂的数据类型(例如结构体),需要保证结构体成员的字节表示没有问题。 - 内存重叠问题:与
memcpy不同,memcmp不会处理内存重叠的情况,它只比较内存的内容,而不考虑是否存在重叠。因此在使用memcmp时要确保比较的内存区域不会重叠。
应用场景:
- 内存区域比较:
memcmp适用于需要比较两个内存区域是否相等的场景,例如在数据校验或加密算法中经常使用它来比较数据块的内容。 - 查找差异:当需要查找两个内存块中不同的位置时,可以使用
memcmp来快速发现差异。
相关文章:
高阶C语言|深入理解字符串函数和内存函数
文章目录 前言1.求字符串长度1.1 字符串长度函数:strlen模拟实现 长度不受限制的字符串函数1.2 字符串拷贝函数:strcpy模拟实现 1.3 字符串连接函数:strcat模拟实现 1.4 字符串比较函数:strcmp模拟实现 长度受限制的字符串函数2.1…...
Pandas DataFrame 拼接、合并和关联
拼接:使用 pd.concat(),可以沿着行或列方向拼接 DataFrame。 合并:使用 pd.merge(),可以根据一个或多个键进行不同类型的合并(左连接、右连接、全连接、内连接)。 关联:使用 join() 方法,通常在设置了索引的 DataFrame 上进行关联操作。 concat拼接 按列拼接 df1 = …...
特种作业操作之低压电工考试真题
1.下面( )属于顺磁性材料。 A. 铜 B. 水 C. 空气 答案:C 2.事故照明一般采用( )。 A. 日光灯 B. 白炽灯 C. 压汞灯 答案:B 3.人体同时接触带电设备或线路中的两相导体时,电流从一相通过人体流…...
“Play around” 在编程领域的含义
“Play around” 的含义 If everything is working correctly we should have a play around in the prompt and verify that functions are actually a new type of value now, not symbols. https://www.buildyourownlisp.com/chapter11_variables 在这段话中,“p…...
[免费]基于Python的Django博客系统【论文+源码+SQL脚本】
大家好,我是java1234_小锋老师,看到一个不错的基于Python的Django博客系统,分享下哈。 项目视频演示 【免费】基于Python的Django博客系统 Python毕业设计_哔哩哔哩_bilibili 项目介绍 随着互联网技术的飞速发展,信息的传播与…...
通过protoc工具生成proto的pb.go文件以及使用protoc-go-inject-tag工具注入自定义标签
1.ProtoBuf认识,安装以及用法 参考:[golang 微服务] 3. ProtoBuf认识,安装以及golang 中ProtoBuf使用 2. 使用protoc-go-inject-tag工具注入自定义标签 这里有一个案例: syntaxproto3; package test;option go_package ".;test";message MyMessage {int6…...
进程池的制作(linux进程间通信,匿名管道... ...)
目录 一、进程间通信的理解 1.为什么进程间要通信 2.如何进行通信 二、匿名管道 1.管道的理解 2.匿名管道的使用 3.管道的五种特性 4.管道的四种通信情况 5.管道缓冲区容量 三、进程池 1.进程池的理解 2.进程池的制作 四、源码 1.ProcessPool.hpp 2.Task.hpp 3…...
Gurobi 基础语法之 tupledict 和 tuplelist
Python中的字典:dict 我们先来介绍一下Python语法中的 dict 类型, 字典中可以通过任意键值来对数据进行映射,任何无法修改的python对象都可以当作键值来使用,这些无法修改的Python对象包括:整数(比如:1),浮…...
Flutter:搜索页,搜索bar封装
view 使用内置的Chip简化布局 import package:chenyanzhenxuan/common/index.dart; import package:ducafe_ui_core/ducafe_ui_core.dart; import package:flutter/material.dart; import package:get/get.dart; import package:tdesign_flutter/tdesign_flutter.dart;import i…...
IoTDB 2025 春节值班与祝福
2025 春节快乐 瑞蛇迎吉庆,祥光映华年,2025 春节已近在眼前。社区祝福 IoTDB 的所有关注者、支持者、使用者 2025 新年快乐,“蛇”来运转! IoTDB 团队的春节放假时间为 2025 年 1 月 27 日至 2 月 4 日,1 月 25 日、26…...
14、Java 对象关系映射(ORM)框架:简化数据库操作的利器
嘿,Java 开发者们!在我们的编程旅程中,经常会遇到一个重要的任务,那就是将 Java 对象和数据库表进行交互。传统的 JDBC 编程虽然强大,但代码往往会变得繁琐且容易出错。这时候,对象关系映射(ORM…...
鲁滨逊漂流记读后感
前言:学校要求出鲁滨逊漂流记的读后感啊,那么今天我就写着试试叭,好久都没更新了嘤,可能写的不好嗷。真的不是很建议参考,因为我的思想可能会与学校的要求不同,更多的是介入了自己的思考,从鲁滨逊好的地方和…...
【面试】【前端】前端网络面试题总结
一、前端网络面试题总结 网络相关知识是前端开发的核心内容之一,面试中通常会考察协议、网络模型、性能优化及常见网络问题的处理。以下是针对前端网络面试题的总结: (一)协议森林(大话网络协议) 网络协议…...
【MQ】如何保证消息队列的高性能?
零拷贝 Kafka 使用到了 mmap 和 sendfile 的方式来实现零拷贝。分别对应 Java 的 MappedByteBuffer 和 FileChannel.transferTo 顺序写磁盘 Kafka 采用顺序写文件的方式来提高磁盘写入性能。顺序写文件,基本减少了磁盘寻道和旋转的次数完成一次磁盘 IO࿰…...
刀客doc:禁令影响下,TikTok广告业务正在被对手截胡
一、 现如今,TikTok在美国的命运迎来了暂时的反转,根据Adage的报道,广告主的投放在恢复。但短暂的关闭带来的影响依然有余震,一些广告主在重新评估TikTok在自己广告预算中的地位,这些是竞争对手截胡的机会。 长期以…...
中国电信AI大模型发布:评分超o1-preview,近屿智能带您探索AI技术新境界
近日,中国电信人工智能研究院宣布,其自主研发的复杂推理大模型TeleAI-t1-preview即将上线天翼AI开放平台。该模型采用强化学习训练方法,显著提升了逻辑推理和数学推导的准确性,展现了强大的复杂问题解决能力。 在权威评测中&#…...
服务定位器模式
服务定位器模式 引言 服务定位器模式(Service Locator Pattern)是一种设计模式,旨在解决应用程序中服务管理的问题。它通过提供一个中央服务注册中心,将服务提供者与服务消费者解耦,从而简化了服务的查找和依赖管理。…...
开发环境搭建-4:WSL 配置 docker 运行环境
在 WSL 环境中构建:WSL2 (2.3.26.0) Oracle Linux 8.7 官方镜像 基本概念说明 容器技术 利用 Linux 系统的 文件系统(UnionFS)、命名空间(namespace)、权限管理(cgroup),虚拟出一…...
AI代理框架:突破LLMs极限的未来之路
标题:“AI代理框架:突破LLMs极限的未来之路” 文章信息摘要: 大型语言模型(LLMs)已接近通过预训练和数据扩展所能达到的极限,未来的AI进步将依赖于强化学习(RL)和代理框架。代理框架…...
Git 如何将旧仓库迁移新仓库中,但不显示旧的提交记录
一、异常错误 场景:我想把旧仓库迁移新仓库中,放进去之后,新仓库会显示这个项目之前的所有提交,如何不显示这些旧的提交? 二、原因 我们需要将旧仓库迁移新仓库中,但是又不想在新仓库中显示旧的提交记录…...
人工智能在计算机视觉中的应用与创新发展研究
一、引言 1.1 研究背景与意义 1.1.1 研究背景 在当今数字化与智能化飞速发展的时代,人工智能已成为推动各领域变革的核心力量,而计算机视觉作为人工智能领域中极具活力与潜力的重要分支,正发挥着日益关键的作用。计算机视觉旨在赋予计算机…...
使用Python和Qt6创建GUI应用程序--关于Qt的一点介绍
关于Qt的一点介绍 Qt是一个免费的开源部件工具包,用于创建跨平台GUI应用程序,允许应用程序从Windows瞄准多个平台,macOS, Linux和Android的单一代码库。但是Qt不仅仅是一个Widget工具箱和功能内置支持多媒体,数据库&am…...
4、PyTorch 第一个神经网络,手写神经网络的基本部分组成
假设有一个二维数据集,目标是根据点的位置将它们分类到两个类别中(例如,红色和蓝色点)。 以下实例展示了如何使用神经网络完成简单的二分类任务,为更复杂的任务奠定了基础,通过 PyTorch 的模块化接口&#…...
挂载mount
文章目录 1.挂载的概念(1)挂载命令:mount -t nfs(2)-t 选项:指定要挂载的文件系统类型(3)-o选项 2.挂载的目的和作用(1)跨操作系统访问:将Windows系统内容挂载到Linux系统下(2)访问外部存储设备(3)整合不同的存储设备 3.文件系统挂载要做的事…...
算法刷题Day30
题目链接 描述 解题思路 考点:动态规划 dp[i][j]表示当前坐标的最小路径和dp初始化状态转移: dp[i][j] matrix[i][j] min(dp[i-1][j],dp[i][j-1]) 比较正上方和正左方的路径和哪个小。取小的那条路 代码 import copy class Solution:def minPathS…...
【R语言】数学运算
一、基础运算 R语言中能实现加、减、乘、除、求模、取整、取绝对值、指数、对数等运算。 x <- 2 y <- 10 # 求模 y %% x # 整除 y %/% x # 取绝对值 abs(-x) # 指数运算 y ^x y^1/x #对数运算 log(x) #log()函数默认情况下以 e 为底 双等号“”的作用等同于identical(…...
DeepSeek助攻!VS Code+Continue 解放双手编程!
简介 要想在vscode中采用AI,那么就需要添加AI插件,通过API来访问不同的模型。 Continue 插件 一款常用的AI代码助手,可以通过vscode和jetbrains来自动补全,推演代码。还有聊天功能。 https://marketplace.visualstudio.com/item…...
当高兴、尊重和优雅三位一体是什么情况吗?
英语单词 disgrace 表示“失脸,耻辱,不光彩,名誉扫地”一类的含义,可做名词或动词使用,含义基本一致,只是词性不同。 disgrace n.丢脸;耻辱;不光彩;令人感到羞耻的人(或…...
日志收集Day008
1.zk集群优化 修改zookeeper的堆内存大小,一般情况下,生产环境给到2G足以,如果规模较大可以适当调大到4G。 (1)配置ZK的堆内存 vim /app/softwares/zk/conf/java.env export JAVA_HOME/sortwares/jdk1.8.0_291 export JVMFLAGS"-Xms2…...
JVM栈溢出线上环境排查
#查看当前Linux系统进程ID、线程ID、CPU占用率(-eo后面跟想要展示的列) ps H -eo pid,tid,%cpups H -eo pid,tid,%cpu |grep tid #使用java jstack 查看进程id下所有线程id的情况 jstack pid 案例2 通过jstack 排查死锁问题 #启动java代码 jstack 进…...
