【再识C进阶3(上)】详细地认识字符串函数、进行模拟字符串函数以及拓展内容

       小编在写这篇博客时,经过了九一八,回想起了祖国曾经的伤疤,勿忘国耻,振兴中华!加油,逐梦少年!

前言

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

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

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

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

学习目标:

       在上一篇博客中,我们已经充分的学习了指针的相关知识,那么这篇博客,我们来重点关注一下字符串函数,主要学习求字符串长度的函数长度受限制的字符串函数长度不收限制的字符串函数,以及字符串查找函数,最后再来介绍一下错误信息报告

       之后,我们会将一些函数进行模拟实现,再拓展一下其他内容,C语言中对字符和字符串的处理很是频繁,但是C语言本身是没有字符串类型的,字符串类型通常放在常量字符串或者字符数组中,字符串常量适用于那些对它不做修改的字符串函数。所以这篇博客还是比较重要的。


学习内容:

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

  1. 求字符串长度的函数,以及手撕代码,模拟实现
  2. 长度受限制的字符串函数,同样手撕代码,模拟实现
  3. 长度不受限制的字符串函数,观看其源代码,了解原理
  4. 字符串查找函数,同样手撕代码,模拟实现
  5. 介绍一下错误信息报告

       小编了解有些人是不会看学习目标,那么我就再说一遍:说起字符串,C语言中对字符和字符串的处理很是频繁,但是C语言本身是没有字符串类型的,字符串通常放在常量字符串中或者字符数组中。常量字符串适用于那些对他不做修改的字符串函数。 

一、求字符串长度的函数——strlen

1.1 函数介绍

       在【初阶C语言3】特别详细地介绍函数中,小编已经介绍了如何查找库函数的参数和具体用法,在这里小编就不多说了。回归主题,我们来看strlen函数。

       简单来说,strlen函数是求字符串长度的库函数,传递的参数是要求字符串长度的首元素地址即可。但是要注意的是strlen函数本质上统计的是字符串中 '\0' 之前的字符的个数。看下面的代码进行区分:

#include <stdio.h>
#include <string.h>  //别忘了引头文件
int main()
{char arr1[] = "sdgeghkoe";char arr2[] = "sdgeg\0hkoe";printf("strlen(arr1) = %d\n", strlen(arr1));printf("strlen(arr2) = %d\n", strlen(arr2));return 0;
}

       在来看strlen函数返回值的特点,仔细看,在库函数中,strlen函数的返回值的类型是无符号整形(size_t)。怎么理解这个无符号整形呢?我们来看下面的一道题:

int main()
{const char* str1 = "abcdfre";  //const修饰,使指针指向的字符串不能被修改const char* str2 = "fsfe";if (strlen(str2) - strlen(str1) > 0){printf("str2 > str1");}else{printf("str1 > str2");}return 0;
}

       这道题按照常理来讲,str2指向的字符串长度比str1指向的字符串长度小,所以这个运算的结果应该是str1 > str2,但为什么结果是str2 > str1,。

       因为strlen函数的返回值是无符号整形两个无符号整形进行相加减的时候是不会出现负数的情况,所以不论这道题的结果是什么,永远都是str2 > str1。

1.2 模拟实现一下strlen函数

       模拟实现strlen函数有三种方法:一个是计数器、一个是递归的方法、一个是指针减指针的方法。下面,小编来一一为大家进行讲解:

1.2.1 计数器方法实现strlen函数

1.3 总结一下strlen函数的要点

二、长度不受限制的字符串函数

2.1 strcpy函数

2.1.1 函数介绍

       strcpy——string copying,表示的是字符串拷贝的工作,如果我们不清楚,我们可以在网站上进行搜索strcpy函数。

       上面的意思是: 将源头的字符串传给目的地的字符串中,包括 '\0'。在使用strcpy函数进行拷贝字符串的时候,源字符串必须要有 '\0'其是strcpy函数拷贝结束的标志,否则程序将会打印出一些不必要的内容。请看下图进行解释:

基本用法:

//第一种写法
int main()
{char arr1[20] = "xxxxxxxxx";char arr2[] = { 's','a','d','\0'};printf("%s\n", strcpy(arr1, arr2));return 0;
}
//第二种写法
int main()
{char arr1[20] = "xxxxxxxxx";char arr2[] = "sad";printf("%s\n", strcpy(arr1, arr2));return 0;
}
int main()
{char arr1[20] = "xxxxxxxxx";char arr2[] = { 's','a','d' };printf("%s\n", strcpy(arr1, arr2));return 0;
}

2.1.2 strcpy函数使用时出现的一些问题

问题一:目的地的字符数组的大小比源头的字符数组的大小要小

代码:

int main()
{char arr1[] = { 0 };char arr2[] = "hello";printf("%s\n", strcpy(arr1, arr2));return 0;
}

结果:

       虽然会弹出警告,但是strcpy函数还是会完成其使命,将源头的字符串拷贝到目的地的字符串中,因为strcpy函数是不受长度限制的函数。

问题二:将源头的字符串拷贝到常量字符串中

代码:

int main()
{char* p = (char* )"feifhw";char arr2[] = "hello";printf("%s\n", strcpy(p, arr2));return 0;
}

结果:

2.1.3 总结一下strcpy函数的要点

  1. 源字符串必须以 '\0' 结束
  2. 在拷贝的过程中,会将源字符串中的 '\0' 拷贝到目标空间中
  3. 目标空间必须足够大,以确保能存放源字符串
  4. 目标空间必须可变,不能是常量字符串

2.1.4 模拟实现一下strcpy函数

char* my_strcpy(char* base, const char* serc) //源头的内容不用改,用const修饰
{assert(base && serc); //检查指针是否指空char* ans = base;while (*base++ = *serc++){;}return ans;
}

2.2 strcat函数

2.2.1 函数介绍

       这个函数是用于字符追加的,如果我们不清楚,我们依然可以进行在网站上查询strcat函数。下面小编我将会为大家用代码示范一下如何使用strcat函数:

基本用法:

int main()
{char arr1[20] = "asd";char arr2[] = "hjk";printf("%s\n", strcat(arr1, arr2));return 0;
}

       在上面的用法中,我们好像已经浅浅地了解了strcat函数的用法,就是将源头的字符串追加到目的地的字符串中。那么这时,我们要来思考一下问题:1)到底是遇见 '\0' 进行追加,还是遇见字符串末尾进行追加呢?2)字符串在拷贝的过程中是否将源头的 '\0' 也拷贝过去呢?3)目的地字符串数组是否要足够大?下面我们来解决一下这些问题:

2.2.2 解决上述strcat函数的相关问题

问题一:到底是遇见什么进行追加(遇见'/0'进行追加)

问题二:在拷贝的过程中将源头的'\0'拷贝过去

int main()
{char arr1[20] = "asdxxx\0xxxxx";char arr2[] = "hjk";printf("%s\n", strcat(arr1, arr2));return 0;
}

问题三:目的地数组要足够大(与strcpy类似)

int main()
{char arr1[] = "asdxxx";char arr2[] = "hjk";printf("%s\n", strcat(arr1, arr2));return 0;
}

2.2.3 总结一下strcat函数的要点

  1. 源字符串必须要以 '\0' 结束,保证能够找到目标空间的末尾
  2. 目标变量必须要足够大,能容纳下源字符串的内容
  3. 目标函数必须可以修改

2.2.4 模拟实现一下strcat函数

char* my_strcat(char* str1, const char* str2)
{assert(str1 && str2);char* ret = str1;//找到目标空间的末尾while (*str1 != 0){str1++;}//数据追加while (*str1++ = *str2++){;}return ret;
}

2.2.5 思考一下字符串能和自己追加呢?

       显然是不能的,因为如果自己给自己进行追加,会导致字符串的内容发生一些变化,将字符0给堵盖掉,就没有结束标志了,看下图进行理解:

2.3 strcmp函数

2.3.1 函数介绍

       strcmp——string compare,表示的是字符串比较的工作,注意这个字符串函数比较的不是字符串的大小,而是对应位置上字符的大小(ASCII值)。我们也可以在网站上搜索strcmp函数。 

标准规定:

  • 第一个字符串大于第二个字符串,则返回大于0的数字
  • 第一个字符串等于第二个字符串,则返回0
  • 第一个字符串小于第二个字符串,则返回小于0的数字

       要注意的是,这个函数的返回值为int,两个参数都是const char*类型的,因为我只是想要比较两个字符串的大小,不需要进行修改。 

基本用法:

int main()
{char arr1[] = "dsgfer";char arr2[] = "dsgfty";printf("%d\n", strcmp(arr1, arr2));return 0;
}

2.3.2 模拟实现一下strcmp函数

//第一种写法,返回值是1和-1
int my_strcmp(const char* str1, const char* str2)
{assert(str1 && str2);while (*str1 == *str2){if (*str2 == 0)return 0;str1++;str2++;}if (*str1 < *str2)return -1;else if(*str1 > *str2)return 1;
}//第二种写法,返回值不一定是1和-1
int my_strcmp(const char* str1, const char* str2)
{assert(str1 && str2);while (*str1 && *str2 && *str1 == *str2){str1++;str2++;}return *str1 - *str2;
}

2.4 长度不受限制的字符串函数的特点

       这些字符串函数都会将直接完成该完成的任务,不会管目的地字符串数组能否放的下,它都会将放入其中,这些函数是简单粗暴的,而我们学的下几个函数不会像这几个函数一样。

三、长度受限制的字符串函数

       我们来看这几个函数strncpy,strncat,strncmp,它们都多了一个n,也就是number。这些函数都会确定几个字符进行该函数的工作,所以说,它们是长度受限制的字符串函数

3.1 strncpy函数

3.1.1 函数介绍

       在和strcpy函数进行比较时,会发现参数部分多了一个num,要和strcpy进行区分一下,所以其作用是拷贝num个字符从源字符串到目标空间中。我们依然可以在网站上搜索strncpy函数。

3.1.2 strncpy函数要注意的要点

要点一:是否在后面拷贝 '\0' ?

       这个函数并没有在后拷贝 '\0' ,可以看下面的代码:

int main()
{char arr1[] = "xxxxxxxxxxx";char arr2[] = "asd";printf("%s\n", strncpy(arr1, arr2, 3));return 0;
}

要点二:如果源字符串的长度小于num,怎么办? 

       如果源字符串的长度小于num,则在拷贝完字符串之后,在目标的后面追加0,直到num个。

int main()
{char arr1[] = "xxxxxxxxxxx";char arr2[] = "asd";printf("%s\n", strncpy(arr1, arr2, 6));return 0;
}

3.2 strncat函数

3.2.1 函数介绍

       这个函数和strcat函数的功能是一样的,我们同样可以在网站上搜索strncat函数。

基本用法:
 

int main()
{char arr1[20] = "hoojp";char arr2[20] = "wqoieh";strncat(arr1, arr2, 4);printf("%s\n", arr1);return 0;
}

3.2.2 strncat函数要注意的要点

       我们要记住strncat函数将几个字符追加到另一个字符串中,会在后面添加一个 '\0'。看下面的代码:

int main()
{char arr1[20] = "hoojp\0xxxxx";char arr2[20] = "wqoieh";strncat(arr1, arr2, 4);printf("%s\n", arr1);return 0;
}

3.3 strncmp函数

       该函数的功能是比较到出现另一个字符不一样或者一个字符串结束或者num个字符全部比较完。看下图的精确含义:

基本用法:

int main()
{char arr1[20] = "hjoihsoad";char arr2[20] = "hjoihjdkp";int ret = strncmp(arr1, arr2, 5);printf("%d\n", ret);return 0;
}

四、字符串查找函数

       这些函数还是比较怪的,不同于前面讲述的三类函数,是一下关于字符串内容的操作,我们来一起看看吧!

4.1 strstr函数

4.1.1 函数介绍

       strstr——string string,是在字符串中查找子字符串中的操作,根据函数原型,可以发现是在str1中找到str2第一次出现的位置,如果str1中没有str2,就返回NULL。

基本用法:

int main()
{char str[] = "This is a simple string";char* pch;pch = strstr(str, "simple");printf("%s", pch);return 0;
}

 

4.1.2 模拟实现一下strstr函数

const char* my_strstr(const char* str1, const char* str2)
{assert(str1 && str2);const char* s1;const char* s2;const char* cp;cp = str1;while (*cp){s1 = cp;s2 = str2;while (*s1 == *s2){if (*s2 == 0)return cp;s1++;s2++;}cp++;}return NULL;
}

4.1.3 拓展一下kmp算法 

4.2 strtok函数

       strtok函数是用来切割字符串的,如果在一个字符串中有分隔符的话,我们可以使用这个函数去按照分隔符将一个字符串分割成好几份。下面来看这个函数的参数:

  • sep参数是一个字符串,定义了用作分隔符的字符的集合;
  • 第一个参数制定一个字符串,它包含了0个或者多个由字符串中的一个或者多个分隔符分割的标记;
  • strtok函数找到str中的下一个标记,并将其用 '\0' 进行结尾,返回一个指向这个标记的指针(注:strtok函数会改变被操作的字符串,所以在使用strtok函数切分的字符串一般都是临时拷贝的内容并且可以修改)
  • strtok函数的一个参数不为NULL,函数将找到str中的第一个标记,strtok函数将保存它在字符串中的位置;
  • strtok函数的一个参数为NULL,函数将在同一个字符串中被保存的位置开始,查找下一个标记;
  • 如果字符串中不存在更多标记,则返回NULL指针。 

基本用法:

int main()
{char arr[] = "shdakio@iahd.isauh";char buf[200] = { 0 };strcpy(buf, arr);const char* p = "@.";char* s = NULL;for (s = strtok(arr, p); s != NULL; s = strtok(NULL, p)){printf("%s\n", s);}return 0;
}

五、错误信息报告函数strerror

       strerror函数是将错误码翻译成错误信息,返回错误信息的字符串的起始位置。错误码有是什么,C语言在使用库函数时,如果发生错误,会将错误码放在errno中,errno是全局变量,可直接使用。

基本用法:

int main()
{int i = 0;for (int i = 0; i < 10; i++){printf("%s\n", strerror(i));}return 0;
}


学习产出:

  1. 求字符串长度的函数,以及手撕代码,模拟实现
  2. 长度受限制的字符串函数,同样手撕代码,模拟实现
  3. 长度不受限制的字符串函数,观看其源代码,了解原理
  4. 字符串查找函数,同样手撕代码,模拟实现
  5. 介绍一下错误信息报告

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

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

相关文章

Linux 链表示例 LIST_INIT LIST_INSERT_HEAD

list(3) — Linux manual page 用Visual Studio 2022创建CMake项目 * CmakeLists.txt # CMakeList.txt : Top-level CMake project file, do global configuration # and include sub-projects here. # cmake_minimum_required (VERSION 3.12)project ("llist")# I…

Remix 2.0 正式发布,现代化全栈Web框架!

9 月 16 日&#xff0c;全栈 Web 框架 Remix 正式发布了 2.0 版本&#xff0c;Remix 团队在发布 1.0 版本后经过近 2 年的持续努力&#xff0c;发布了 19 个次要版本、100 多个补丁版本&#xff0c;并解决了数千个问题和拉取请求&#xff0c;终于迎来了第二个主要版本&#xff…

【数据结构】二叉搜索树与Map和Set

目录 ♫二叉搜索树 ♪什么是二叉搜索树 ♪二叉搜索树的特性 ♪模拟实现二叉搜索树 ♫Map ♪什么是Map ♪Map的内部类 ♪Map的常用方法 ♪Map的遍历 ♫Set ♪什么是Set ♪Set的常用方法 ♪Set的遍历 ♫二叉搜索树 ♪什么是二叉搜索树 二叉搜索树又称二叉排序树&#…

多线程带来的的风险-线程安全

多线程带来的的风险-线程安全 ~~ 多线程编程中,最难的地方,也是一个最重要的地方&#xff0c;还是一个最容易出错的地方,更是一个面试中特别爱考的地方.❤️❤️❤️ 线程安全的概念 万恶之源,罪魁祸首是多线程的抢占式执行,带来的随机性.~~&#x1f615;&#x1f615;&…

14:00面试,14:06就出来了,问的问题过于变态了。。。

从小厂出来&#xff0c;没想到在另一家公司又寄了。 到这家公司开始上班&#xff0c;加班是每天必不可少的&#xff0c;看在钱给的比较多的份上&#xff0c;就不太计较了。没想到5月一纸通知&#xff0c;所有人不准加班&#xff0c;加班费不仅没有了&#xff0c;薪资还要降40%…

redis安装问题

title: “Redis安装问题” createTime: 2022-01-04T20:47:0608:00 updateTime: 2022-01-04T20:47:0608:00 draft: false author: “name” tags: [“redis”] categories: [“install”] description: “测试的” title: redis安装可能遇到的错误 createTime: 2022-01-04T20:47…

第52节:cesium 3DTiles模型特效+选中高亮(含源码+视频)

结果示例: 完整源码: <template><div class="viewer"><vc-viewer @ready="ready" :logo="false"><vc-navigation

【【萌新的FPGA学习之实战流水灯】】

萌新的FPGA学习之实战流水灯 实验任务 本节的实验任务是使用领航者底板上的两个 PL LED 灯顺序点亮并熄灭&#xff0c;循环往复产生流水灯的效 果&#xff0c;流水间隔时间为 0.5s。 1MHz&#xff1d;1000000Hz 10的6次方 1ns&#xff1d;10的-9次方秒 开发板晶振50Mhz 计算得…

自主研究,开发并产业化的一套UWB精确定位系统源码 UWB源码

UWB (ULTRA WIDE BAND) 技术是一种无线载波通讯技术&#xff0c;它不采用正弦载波&#xff0c;而是利用纳秒级的非正弦波窄脉冲传输数据&#xff0c;因此其所占的频谱范围很宽。UWB定位系统研发团队依托在移动通信&#xff0c;雷达&#xff0c;微波电路&#xff0c;云计算与大数…

全国职业技能大赛云计算--高职组赛题卷②(容器云)

全国职业技能大赛云计算--高职组赛题卷②&#xff08;容器云&#xff09; 第二场次题目&#xff1a;容器云平台部署与运维任务1 Docker CE及私有仓库安装任务&#xff08;5分&#xff09;任务2 基于容器的web应用系统部署任务&#xff08;15分&#xff09;任务3 基于容器的持续…

Qt/C++音视频开发55-加密保存到文件并解密播放

一、前言 为了保证视频文件的安全性&#xff0c;有时候需要对保存的视频文件加密&#xff0c;然后播放的时候解密出来再播放&#xff0c;只有加密解密的秘钥一致时才能正常播放&#xff0c;用ffmpeg做视频文件的加密保存和解密播放比较简单&#xff0c;基于ffmpeg强大的字典参…

FPGA:卷积编码及维特比译码仿真

FPGA&#xff1a;卷积编码及维特比译码仿真 本篇记录一下在FPGA中完成卷积编码和维特比译码的过程&#xff0c;通过代码解释编码的过程和译码的过程&#xff0c;便于理解&#xff0c;同时也方便移植到其他工程中。 1. 准备工作 卷积编译码IP核—convolutionIP核和viterbiIP核…

STM32F407 串口使用DMA方式通信

DMA的原理&#xff0c;就是利用寄存器方式进行读写&#xff0c;这样的好处就是相对于中断触发&#xff08;往往一个字节字节的就中断一次&#xff09;&#xff0c;CPU中断次数大大降少&#xff0c;提高了效率&#xff0c;但也影响了实时性。总体来说&#xff0c;对于一般的应用…

Oracle 12c自动化管理特性的新进展:自动备份、自动恢复和自动维护功能的优势|oracle 12c相对oralce 11g的新特性(3)

一、前言: 前面几期讲解了oracle 12c多租户的使用、In-Memory列存储来提高查询性能以及数据库的克隆、全局数据字典和共享数据库资源的使用 今天我们讲讲oracle 12c的另外的一个自动化管理功能新特性:自动备份、自动恢复、自动维护的功能 二、自动备份、自动恢复、自动维护…

Android开发笔记 :理解Fragment

Android开发笔记&#xff1a;理解Fragment 导言 本篇文章产生的原因很简单&#xff0c;就是我在了解Android Jetpack中的Lifecycle框架时发现Lifecycle具体时间和状态的更新都是由一个名为ReportFragment的Fragment来跟踪的&#xff0c;为了更好的了解Fragment是如何追踪Activ…

机器学习的主要内容

分类任务 回归任务 有一些算法只能解决回归问题有一些算法只能解决分类问题有一些算法的思路既能解决回归问题&#xff0c;又能解决分类问题 一些情况下&#xff0c; 回归任务可以转化为分类任务&#xff0c; 比如我们预测学生的成绩&#xff0c;然后根据学生的成绩划分为A类、…

LeetCode刷题

一 【移除元素】 原题链接&#xff1a;27. 移除元素 - 力扣&#xff08;LeetCode&#xff09; 给你一个数组 nums 和一个值 val&#xff0c;你需要 原地 移除所有数值等于 val 的元素&#xff0c;并返回移除后数组的新长度。 不要使用额外的数组空间&#xff0c;你必须仅使用…

基因组注释(Annotation)

基因组组装完成后&#xff0c;或者是完成了草图&#xff0c;就不可避免遇到一个问题&#xff0c;需要对基因组序列进行注释。注释之前首先得构建基因模型&#xff0c;有三种策略&#xff1a; 从头注释(de novo prediction)&#xff1a;通过已有的概率模型来预测基因结构&#…

C++17中std::filesystem::path的使用

C17引入了std::filesystem库(文件系统库, filesystem library)。这里整理下std::filesystem::path的使用。 std::filesystem::path&#xff0c;文件系统路径&#xff0c;提供了对文件系统及其组件(例如路径、常规文件和目录)执行操作的工具。此path类主要用法包括&#x…

【Kafaka实现高吞吐量、低延迟的底层原理】

文章目录 Kafaka实现高吞吐量、低延迟的底层原理顺序写入Page Cache零拷贝分区分段索引批量读写批量压缩 Kafaka实现高吞吐量、低延迟的底层原理 Kafka虽然是基于磁盘做的数据存储&#xff0c;但却具有高并发、高吞吐量、低延时的特点&#xff0c;其吞吐量动辄几万、几十上百万…