当前位置: 首页 > news >正文

常见的字符串函数(包含头文件string.h)和字符函数(2)

八. strstr函数

1.strstr的定义
char *strstr( const char *str1, const char *str2 );

->1. strstr查找子串(str2)在字符串(str2)中第一次出现的位置,记录并返回该位置的指针,如果找不到,则返回NULL

->2. str1:查找字符串的目标空间    str2:需要查找的对象字符串

->3. 因为strstr函数操作时不会改变形参的指向,所以我们在两个形参前面加上两个const

2.strstr的使用

例1:

记录子串在目标字符串中的位置,然后返回子串首地址

#include <stdio.h>
#include <string.h>void main(void)
{char* str1 = "Strive to improve yourself";char* str2 = "improve";char* det = strstr(str1, str2);if(ret == NULL){printf("字符串不存在!");}else{printf("%s", det);}
}

运行结果:

例2:

返回子串首地址在目标字符串中的具体位置(地址 - 地址的方法)

#include <stdio.h>
#include <string.h>void main(void)
{char* str1 = "Strive to improve yourself";char* str2 = "improve";char* det = strstr(str1, str2);int result = det - str1 + 1;if (det == NULL){printf("字符串不存在!");}else{printf("%d", result);}
}

运行结果:

例3:

记录目标空间中一共出现多少次子串

#include <stdio.h>
#include <string.h>void main(void)
{char* str1 = "Strive to improve improve yourself";char* str2 = "improve";char* p = str1;int count = 0;while (p = strstr(p, str2)){count++;p++;}printf("%d", count);
}

运行结果:

3.strstr的模拟实现
#include <stdio.h>
#include <assert.h>char* My_strstr(const char* str1, const char* str2)
{assert(str1 && str2);char* p = str1;char* s1 = str1;char* s2 = str2;while (*p){s1 = p;s2 = str2;while (*s1 != '\0' && *s2 != '\0' && *s1 == *s2){s1++;s2++;}if (*s2 == '\0')return p;p++;}return NULL;
}int main()
{char* str1 = "abbbcdf";char* str2 = "bbc";const char* det = My_strstr(str1, str2);if (det == NULL){printf("字符串不存在!");}else{printf("%s", det);}return 0;
}

九. strtok函数

1.strtok的定义
char *strtok( char *strToken, const char *strDelimit );

->1. 作用:切割字符串。将分隔符转换成\0在将前面那个字符串的首地址返回给函数(返回\0前面的字符串)

->2. char *strToken  目标空间

->3. char  *strDelimit 标记(分隔符)

2.strtok的使用

例1:

#include <stdio.h>
#include <string.h>int main()
{const char* sep = "@.";//三个分隔符 @ . \0 别忘了字符串还带有\0char email[] = "zhangsan@@nianxi.nbaiwan";char cp[30] = { 0 };   strcpy(cp, email);  char *ret = strtok(cp, sep);if(ret != NULL){printf("%s ", ret);}return 0;
}

因为strtok函数会改变原字符串中的内容,所以一般都是使用临时拷贝的内容,并且可修改

运行结果:

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

为NULL,函数将在同一个字符串中被保存的位置开始,查找下一个标记

例2:

#include <stdio.h>
#include <string.h>int main()
{const char* sep = "@.";char email[] = "zhangsan@@nianxi.nbaiwan";char cp[30] = { 0 };   strcpy(cp, email);  char *ret = strtok(cp, sep);if(ret != NULL){printf("%s\n", ret);}ret = strtok(NULL, sep);if(ret == NULL){printf(第二次没有找到该标记);}else{printf("%s\n", ret);}return 0;
}

如果两个标记在一起中间什么都没有,函数strtok就啥也获取不到,获取不到strtok会直接跳过

不管它,直接去找下一个标记

运行结果:

例3:

第三次分割

#include <stdio.h>
#include <string.h>int main()
{const char* sep = "@.";/* char email[] = "zhangsan@.nianxinbaiwan";*/char email[] = "zhangsan@@nianxi.nbaiwan";char cp[30] = { 0 };strcpy(cp, email);char* ret = strtok(cp, sep);if (ret != NULL){printf("%s\n", ret);}ret = strtok(NULL, sep);if (ret == NULL){printf("第二次没有找到该标记");}else{printf("%s\n", ret);}ret = strtok(NULL, sep);if (ret == NULL){printf("第三次什么也没获取到");}else{printf("%s\n", ret);}return 0;
}

运行结果:

例4:

如果strtok什么都没找到就会返回NULL

第四次从字符串末尾开始向后找,后面已经没有字符串了,所以找不到

#include <stdio.h>
#include <string.h>int main()
{const char* sep = "@.";/* char email[] = "zhangsan@nianxin.baiwan";*/char email[] = "zhangsan@@nianxi.nbaiwan";char cp[30] = { 0 };strcpy(cp, email);char* ret = strtok(cp, sep);if (ret != NULL){printf("%s\n", ret);}ret = strtok(NULL, sep);if (ret == NULL){printf("第二次没有找到该标记\n");}else{printf("%s\n", ret);}ret = strtok(NULL, sep);if (ret == NULL){printf("第三次什么也没获取到\n");}else{printf("%s\n", ret);}ret = strtok(NULL, sep);if (ret == NULL){printf("第四次什么也没获取到\n");}else{printf("%s\n", ret);}return 0;
}

运行结果:

因为每次都需要用if else来判断,有点重复不好看

所以我们将代码改进一下:

利用for循环

#include <stdio.h>
#include <string.h>int main()
{const char *sep = "@.";char email[] = "zhangsan@nianxin.baiwan";char pc[30] = { 0 };strcpy(pc, email);char *ret =0;ret = strtok(pc, sep);for(; ret != NULL; ret = strtok(NULL, sep)){printf("%s\n", ret);}return 0;
}
3.strtok的模拟实现

例1:

#include <stdio.h>
#include <string.h>
#include <assert.h>char* My_strtok(char* ps, const char* pc)
{assert(pc); //不用加上ps因为第二次用的话会传参NULLstatic char* str1 = NULL;static char* str2 = NULL;static int count = 0;static int sz1 = 0; int sz2 = 0;if(ps != NULL){str1 = ps;sz1 = strlen(ps);sz2 = strlen(pc);for(*ps; *ps != '\0'; ps++){for(int i = 0; i < sz2; i++){if(i == 0){count++;}if(*ps == *(pc + i)){*ps = '\0';str2 = ps;return str1;}}}else{str1 = str2 + 1;ps = str1;for(*ps; *ps = '\0'; ps++){for(int i = 0; i < sz2; i++){if(i == 0){count++;}if(*ps == *(pc + i)){ps = '\0';str2 = psreturn str1;}}}if(count == sz1){return NULL;}return str1;}
}
int main()
{const char* pc ="b";char email[] ="nianxinbaiwan";char ps[30] = { 0 };strcpy(ps, email);char* ret = My_strtok(ps, pc);for(; ret != NULL; ret = My_strtok(NULL, pc)){printf("%s\n", ret);}return 0;
}

十. strerror函数

1.strerror的定义
char *strerror( int errnum );

->1.返回值是字符型的指针

->2.int errnum 错误码:C语言的数据库,在执行失败的时候,都会自动设置错误码

0:No error
1:Operation not permitted
2:No such file or directory
3:No such process
4:Interrupted function call
5:Input/output error
6:No such device or address
7:Arg list too long
8:Exec format error
9:Bad file descriptor

errno - C语言设置的一个全局的错误码存放的变量

2.strerror的使用
#include <stdio.h>
#include <errno.h>int main()
{FILE* pf = fopen("idea.txt", "w");if(pf == NULL){printf("%d", strerror(errno));return 1;}fputc("w");fclose(pf);pf = NULL;return 0;
}    

运行结果:

十一. 字符分类函数

包含头文件 ctype

1.isspace -> 判断是否是空白字符,是就返回非0,不是就返回0

int a = isspace(' ');
printf("%d", a);

2.isdigit -> 判断是否是数字字符,是就返回非0,不是就返回0

 int a = isdigit('x');printf("%d", a);

3.iscntrl -> 任何控字符

4.isxdigit -> 十六进制数字,包括十进制数字,小写字母a - f,大写字母A - F

5.islower -> 小写字母a - z

6.isupper 大写字母A - Z

7.isalpha 字母a - z或A - Z

8.isalnum 字母或数字,a - z,A - Z,0 - 9

9.ispunct 标点符号,任何不属于数字或者字母的图形字符(可打印)

10.isgraph 任何图像字符

11.isprint 任何打印字符,包括图形字符和空白字符

十二. tolower函数

1. tolower的使用
printf("%c\n", tolower('W'));

2.tosupper -> 转大写

十三. memcpy

1.memcpy的定义
void* memcpy(void* destination, const void* source, size_t num);

万能copy函数,什么类型的都可以拷贝

memcpy负责拷贝两块独立的空间中的数据

->1. void* destination 目标空间

->2. void* source 源空间

->3. size_t num 源空间总大小

2.memcpy的使用

例1:

整型拷贝

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

运行结果:

例1:

浮点型拷贝

#include <stdio.h>
#include <string.h>int main()
{float arr1[] = { 1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, };float arr2[10] = { 0.0 };memcpy(arr2, arr1, 28);int i = 0;for (i = 0; i < sizeof(arr2) / sizeof(arr2[0]); i++){printf("%d ", arr2[i]);}return 0;
}

运行结果:

tips:

strcpy和memcpy的区别:

1.复制内容不同,strcpy只能复制字符类型,而memcpy可以复制任意类型

2.所需参数不同,strcpy只需要目标空间和源空间,而memcpy在次之上还需要源空间总大小

3.用途不同,字符类的复制用strcpy,其他类型的复制使用memcpy

3.memcpy的模拟实现

我们写的这个模拟实现的memcpy是不能实现重叠空间的拷贝的

#include <stdio.h>
#include <assert.h>void* My_memcpy(void* det, const void* src, size_t num)
{assert(det&&src);void* p = det;while(num--){*(char*)det = *(char*)src;det = (char*)det + 1;src = (char*)src + 1; }return p;
}

 为什么不使用(char*)src++,(char*)det++这种写法,因为这种写法有点问题,有些编译器是不支持这样写的,被(char*)强转了就是一个临时的变量,对一个临时的变量进行操作是有问题的

用自定义函数My_memcpy进行重复空间的拷贝

#include <stdio.h>
#include <assert.h>void* My_memcpy(void* det, const void* src, size_t num)
{assert(det && src);void* p = det;while (num--){*(char*)det = *(char*)src;det = (char*)det + 1;src = (char*)src + 1;}return p;
}
void test()
{int a[] = { 1,2,3,4,5,6,7,8,9,10 };//int b[20] = { 0 };My_memcpy(a + 2, a, 28);int i = 0;for (i = 0; i < 10; i++){printf("%d ", a[i]);} 
}
int main()
{test();return 0;
}

运行结果:

由上图知:第三第四元素地址被第一第二元素地址覆盖了,后面拿第三第四元素内容和第一第二元素一样,后面的地址跟前面的同理,就会出现如图这样的结果

tips:

上面说了,memcpy用于单独的两个空间(重叠的内存也能实现,但是C中规定它只能用于单独的两个空间),那重叠内存的拷贝应该用什么函数  - > memmove

十四. memmove函数

1.memmove的定义
void *memmove( void *dest, const void *src, size_t count );

负责拷贝重复空间中的数据

->1. void* destination 目标空间

->2. void* source 源空间

->3. size_t num 需要拷贝的字节数

它的目标空间和源空间是同一个空间

2.memmove的使用
#include <stdio.h>
#include <string.h>int main()
{int a[] = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};memmove(a + 2, a, 20);int i = 0;for(i = 0; i < 10; i++){printf("%d ", a[i]);}return 0;
}

运行结果:

3.memmove的模拟实现
#include <stdio.h>
#include <assert.h>void* My_memmove(void* det, const void* src, size_t num)
{assert(det&&src);void* p = det;if (det < src){while (num--){*(char*)det = *(char*)src;det = (char*)det + 1;src = (char*)src + 1;}}else{while (num--){*((char*)det + num) = *((char*)src + num);}}return p;
}
int main()
{int a[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };My_memmove(a + 1, a + 2, 20);int i = 0;for (i = 0; i < 10; i++){printf("%d ", a[i]);}return 0;
}

运行结果:

相关文章:

常见的字符串函数(包含头文件string.h)和字符函数(2)

八. strstr函数 1.strstr的定义 char *strstr( const char *str1, const char *str2 ); ->1. strstr查找子串(str2)在字符串(str2)中第一次出现的位置&#xff0c;记录并返回该位置的指针&#xff0c;如果找不到&#xff0c;则返回NULL ->2. str1&#xff1a;查找字符…...

Python | Leetcode Python题解之第187题重复的DNA序列

题目&#xff1a; 题解&#xff1a; L 10 bin {A: 0, C: 1, G: 2, T: 3}class Solution:def findRepeatedDnaSequences(self, s: str) -> List[str]:n len(s)if n < L:return []ans []x 0for ch in s[:L - 1]:x (x << 2) | bin[ch]cnt defaultdict(int)for…...

SpringCloud分布式微服务链路追踪方案:Skywalking

一、引言 随着微服务架构的广泛应用&#xff0c;系统的复杂性也随之增加。在这种复杂的系统中&#xff0c;应用通常由多个相互独立的服务组成&#xff0c;每个服务可能分布在不同的主机上。微服务架构虽然提高了系统的灵活性和可扩展性&#xff0c;但也带来了新的挑战&#xf…...

首次线下联合亮相!灵途科技携手AEye、ATI亮相2024 EAC 易贸汽车产业大会

6月22日&#xff0c;2024 EAC 易贸汽车产业大会在苏州国际博览中心圆满落幕&#xff0c;泛自动驾驶领域光电感知专家灵途科技携手自适应高性能激光雷达解决方案全球领导者AEye公司&#xff08;NASDAQ:LIDR&#xff09;及光电器件规模化量产巨头Accelight Technologies&#xff…...

一文入门CMake

我们前几篇文章已经入门了gcc和Makefile&#xff0c;现在可以来玩玩CMake了。 CMake和Makefile是差不多的&#xff0c;基本上是可以相互替换使用的。CMAke可以生成Makefile&#xff0c;所以本质上我们还是用的Makefile&#xff0c;只不过用了CMake就不用再写Makefile了&#x…...

【LeetCode面试经典150题】117. 填充每个节点的下一个右侧节点指针 II

一、题目 117. 填充每个节点的下一个右侧节点指针 II - 力扣&#xff08;LeetCode&#xff09; 给定一个二叉树&#xff1a; struct Node {int val;Node *left;Node *right;Node *next; } 填充它的每个 next 指针&#xff0c;让这个指针指向其下一个右侧节点。如果找不到下一个…...

RTDETR更换优化器——Lion

RTDETR更换Lion优化器 论文&#xff1a;https://arxiv.org/abs/2302.06675 代码&#xff1a;https://github.com/google/automl/blob/master/lion/lion_pytorch.py 简介&#xff1a; Lion优化器是一种基于梯度的优化算法&#xff0c;旨在提高梯度下降法在深度学习中的优化效果…...

Spring Boot中最佳实践:数据源配置详解

Spring Boot中最佳实践&#xff1a;数据源配置详解 大家好&#xff0c;我是免费搭建查券返利机器人省钱赚佣金就用微赚淘客系统3.0的小编&#xff0c;也是冬天不穿秋裤&#xff0c;天冷也要风度的程序猿&#xff01;今天我们将深入探讨在Spring Boot中如何进行最佳实践的数据源…...

第1章 物联网模式简介---独特要求和体系结构原则

物联网用例的独特要求 物联网用例往往在功耗、带宽、分析等方面具有非常独特的要求。此外&#xff0c;物联网实施的固有复杂性&#xff08;一端的现场设备在计算上受到挑战&#xff0c;另一端的云容量几乎无限&#xff09;迫使架构师做出艰难的架构决策和实施选择。可用实现技…...

数据挖掘概览

数据挖掘(Data Mining)就是从大量的,不完全的,有噪声的,模糊的,随机的实际应用数据中,提取隐含在其中的,人们事先不知道的,但又是潜在有用的信息和知识的过程. 预测性数据挖掘 分类 定义&#xff1a;分类就是把一些新的数据项映射到给定类别中的某一个类别 分类流程&#x…...

【学习】软件测试中常见的文档类型及其作用

在软件开发的生命周期中&#xff0c;软件测试是确保产品质量的关键步骤。为了系统地进行测试活动&#xff0c;并保证测试结果的有效性和可追溯性&#xff0c;产生了一系列标准化的测试文档。这些文档不仅为测试人员提供了执行指南&#xff0c;而且为项目管理者和利益相关者提供…...

electron的托盘Tray

1.在主进程文件background.js中引入需要的文件 import { Tray, Menu } from "electron"; const path require("path");2.获取托盘图标 const baseSRC process.cwd(); //这里不能使用__dirname,使用dirname会直接获取dist_electron中的文件&#xff0c;…...

Harmony OS UI框架探索笔记

本文探讨了如何将现有的常用架构理论与Arkts和ArkUI结合起来&#xff0c;使代码更有条理&#xff0c;并利用Previewer快速调整布局&#xff0c;同时在不改变代码的情况下运行显示真实数据。 开发环境 Windows 11DevEco Studio 4.0 ReleaseBuild Version: 4.0.0.600, built on…...

transformers evaluate

☆ Evaluate https://huggingface.co/docs/evaluate/main/en/installation ★ 解决方案 常用代码 # 查看支持的评估函数 evaluate.list_evaluation_modules(include_communityTrue)# 加载评估函数 accuracy evaluate.load("accuracy")# load function descripti…...

【ONLYOFFICE深度探索】:ONLYOFFICE桌面编辑器8.1震撼发布,打造高效办公新境界

文章目录 一、功能完善的PDF编辑器&#xff1a;解锁文档处理新维度二、幻灯片版式设计&#xff1a;释放创意&#xff0c;打造专业演示三、改进从右至左显示&#xff1a;尊重多元文化&#xff0c;优化阅读体验四、新增本地化选项&#xff1a;连接全球用户&#xff0c;跨越语言障…...

C++系统相关操作4 - 获取CPU(指令集)架构类型

1. 关键词2. sysutil.h3. sysutil.cpp4. 测试代码5. 运行结果6. 源码地址 1. 关键词 关键词&#xff1a; C 系统调用 CPU架构 指令集 跨平台 实现原理&#xff1a; Unix-like 系统: 可以通过 uname -m 命令获取 CPU 架构类型。Windows 系统: 可以通过环境变量 PROCESSOR_A…...

whisper 实现语音转文字

准备需要转码的音频 https://support.huaweicloud.com/sdkreference-sis/sis_05_0039.html 编码转吗的代码 import whisperif __name__ "__main__":file_path "16k16bit.wav"model whisper.load_model("small")result model.transcribe(f…...

使用VLLM部署llama3量化版

1.首先去魔塔社区下载量化后的llama3模型 git clone https://www.modelscope.cn/huangjintao/Meta-Llama-3-8B-Instruct-AWQ.git 2.跑起来模型 1&#xff09;python -m vllm.entrypoints.openai.api_server --model /home/cxh/Meta-Llama-3-8B-Instruct-AWQ --dtype auto --…...

计算机缺失OpenCL.dll怎么办,OpenCL.dll丢失的多种解决方法

在使用电脑的过程中&#xff0c;我们经常会遇到一些开机弹窗问题。其中&#xff0c;开机弹窗找不到OpenCL.dll是一种常见的情况。本文将详细介绍开机弹窗找不到OpenCL.dll的原因分析、解决方法以及预防措辞&#xff0c;帮助大家更好地解决这一问题。 一&#xff0c;了解OpenCL.…...

git 本地代码管理

简介 git 能实现本地代码多个更改版本的管理和导出。 首先复制好项目&#xff08;参考 git clone 别人项目后正确的修改和同步操作 中的前三步&#xff09; 实操 克隆原始项目 首先&#xff0c;从远程仓库克隆项目到本地&#xff1a; git clone https://github.com/libo-huan…...

springboot 百货中心供应链管理系统小程序

一、前言 随着我国经济迅速发展&#xff0c;人们对手机的需求越来越大&#xff0c;各种手机软件也都在被广泛应用&#xff0c;但是对于手机进行数据信息管理&#xff0c;对于手机的各种软件也是备受用户的喜爱&#xff0c;百货中心供应链管理系统被用户普遍使用&#xff0c;为方…...

【Java学习笔记】Arrays类

Arrays 类 1. 导入包&#xff1a;import java.util.Arrays 2. 常用方法一览表 方法描述Arrays.toString()返回数组的字符串形式Arrays.sort()排序&#xff08;自然排序和定制排序&#xff09;Arrays.binarySearch()通过二分搜索法进行查找&#xff08;前提&#xff1a;数组是…...

【机器视觉】单目测距——运动结构恢复

ps&#xff1a;图是随便找的&#xff0c;为了凑个封面 前言 在前面对光流法进行进一步改进&#xff0c;希望将2D光流推广至3D场景流时&#xff0c;发现2D转3D过程中存在尺度歧义问题&#xff0c;需要补全摄像头拍摄图像中缺失的深度信息&#xff0c;否则解空间不收敛&#xf…...

【RockeMQ】第2节|RocketMQ快速实战以及核⼼概念详解(二)

升级Dledger高可用集群 一、主从架构的不足与Dledger的定位 主从架构缺陷 数据备份依赖Slave节点&#xff0c;但无自动故障转移能力&#xff0c;Master宕机后需人工切换&#xff0c;期间消息可能无法读取。Slave仅存储数据&#xff0c;无法主动升级为Master响应请求&#xff…...

ArcGIS Pro制作水平横向图例+多级标注

今天介绍下载ArcGIS Pro中如何设置水平横向图例。 之前我们介绍了ArcGIS的横向图例制作&#xff1a;ArcGIS横向、多列图例、顺序重排、符号居中、批量更改图例符号等等&#xff08;ArcGIS出图图例8大技巧&#xff09;&#xff0c;那这次我们看看ArcGIS Pro如何更加快捷的操作。…...

Typeerror: cannot read properties of undefined (reading ‘XXX‘)

最近需要在离线机器上运行软件&#xff0c;所以得把软件用docker打包起来&#xff0c;大部分功能都没问题&#xff0c;出了一个奇怪的事情。同样的代码&#xff0c;在本机上用vscode可以运行起来&#xff0c;但是打包之后在docker里出现了问题。使用的是dialog组件&#xff0c;…...

C++.OpenGL (14/64)多光源(Multiple Lights)

多光源(Multiple Lights) 多光源渲染技术概览 #mermaid-svg-3L5e5gGn76TNh7Lq {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-3L5e5gGn76TNh7Lq .error-icon{fill:#552222;}#mermaid-svg-3L5e5gGn76TNh7Lq .erro…...

Java + Spring Boot + Mybatis 实现批量插入

在 Java 中使用 Spring Boot 和 MyBatis 实现批量插入可以通过以下步骤完成。这里提供两种常用方法&#xff1a;使用 MyBatis 的 <foreach> 标签和批处理模式&#xff08;ExecutorType.BATCH&#xff09;。 方法一&#xff1a;使用 XML 的 <foreach> 标签&#xff…...

Python基于历史模拟方法实现投资组合风险管理的VaR与ES模型项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档&#xff09;&#xff0c;如需数据代码文档可以直接到文章最后关注获取。 1.项目背景 在金融市场日益复杂和波动加剧的背景下&#xff0c;风险管理成为金融机构和个人投资者关注的核心议题之一。VaR&…...

消防一体化安全管控平台:构建消防“一张图”和APP统一管理

在城市的某个角落&#xff0c;一场突如其来的火灾打破了平静。熊熊烈火迅速蔓延&#xff0c;滚滚浓烟弥漫开来&#xff0c;周围群众的生命财产安全受到严重威胁。就在这千钧一发之际&#xff0c;消防救援队伍迅速行动&#xff0c;而豪越科技消防一体化安全管控平台构建的消防“…...