【C语言】——字符串函数的使用与模拟实现(上)

【C语言】——字符串函数

前言

  在我们学习 C语言 的过程中,对库函数的使用是必不可少的。其中,相信大家最熟悉的就是 < s t d i o . h > <stdio.h> <stdio.h> 中的 p r i n t print print s c a n f scanf scanf 函数了吧。
  但是今天,我们不讲他们(太难了,呜呜呜),今天我们来讲与字符串相关的函数,也就是 < s t r i n g . h string.h string.h> 中的系列函数。
  要知道,除了输入输出,我们对字符串的操作也是必不可少的,那么了解字符串相关的函数的重要性就不言而喻了,让我们开启接下来的学习吧。

一、 s t r l e n strlen strlen 函数

1.1、函数功能

  • 函数声明:size_t strlen(const char* str);
  • s t r l e n strlen strlen 函数用于求字符串的长度,使用时必须包含头文件
  • 基本原理是:从传递的指针变量开始,从前往后计算字符的个数,直到遇到 ‘\0’ 停止‘\0’ 本身不计算),因此参数指向的字符串必须要以 ‘\0’ 结束
  • 返回值为字符串的长度,注意:返回值是 size_t 类型(易错)

  

1.2、函数的使用

  
   s t r l e n strlen strlen 函数的使用非常简单,这里就不过多介绍了,大家直接看代码

#include<string.h>
#include<stdio.h>int main()
{char* a = "abcdef";int sz = strlen(a);printf("%d\n", sz);return 0;
}


运行结果
在这里插入图片描述

1.3、函数的模拟实现

(1)计数法

  首先,我们要知道指针存储的是字符串首字符的地址。该方法的思路是将传入的字符指针不断往后移每移一次计数器加一,当指针指向 ‘\0’ 时,结束计数,并返回计数的值。
  
代码实现

#include<assert.h>size_t my_strlen(const char* str)
{//assert断言,防止传入空指针assert(str);int count = 0;//判断条件:指针指向的字符是否为0('\0')while (*str){count++;str++;}return count;
}

(2)递归法

  
  有没有办法在求出字符串长度的同时不创建临时变量呢?我们可以试试递归的思想

  那用递归的思想来计算字符串的长度,解题思路是什么呢?

  我们不妨想:比如要计算 a b c abc abc 的长度,是不是可以将他看成 1 + 1+ 1+ b c bc bc 的长度,而 b c bc bc 的长度,又可以将他看成是 1+“ c c c 的长度。
  

在这里插入图片描述

  怎么样,你想到了吗?
  
代码实现

#include<assert.h>
size_t my_strlen(const char* str)
{assert(str);if (!*str)//逻辑取反,当*str为0时为假,逻辑取反为真return 0;elsereturn 1 + my_strlen(str + 1);
}

  

(3)指针 - 指针

  
  还有一个方法就是指针 - 指针,首先我们知道,指针 - 指针,代表的是两指针之间的元素个数(详情请看:【C语言】—— 指针一 : 初识指针(上))
  
  我们可以利用这个思想,创建一个指针变量,让他不断移动,直至指向 '\0' ,再通过两个指针相减,得到字符串的长度。
  
代码实现

#include<assert.h>size_t my_strlen(const char* str)
{assert(str);char* p = str;while (*p++){;}return p - str - 1;
}

代码解析:
  
  while (*arr++);可能这句代码有些小伙伴不太了解,没关系,我们一起来看看

  • *arr++:这里有两个操作符: ∗ * 后置++。 虽然后置++ 的优先级比 ∗ * 高,但是他是后置的,所以先进行解引用
  • 解引用完成后,结果为真进入循环,为假退出循环
  • 无论 * a r r arr arr 结果为真为假,接下来 a r r arr arr 进行自增操作。注:++ 操作数是 a r r arr arr,而非 * a r r arr arr,如果想让 * a r r arr arr 自增,应写为:*(arr)++
  • 因为循环不需要再执行任何操作,因此放置空语句
  • return arr - str - 1;因为前面循环的判断条件不论真假, a r r arr arr 都会自增,即指针指向 ‘\0’ 后依然会往后移动一位,此时如果直接arr - str,则把 ‘\0’ 也计算在内,而 s t r l e n strlen strlen 是不计算
    ‘\0’ 的,因此最终要 - 1

  

二、 s t r c p y strcpy strcpy 函数

2.1、函数功能

在这里插入图片描述

功能 s t r c p y strcpy strcpy 函数实现的是字符串的拷贝,将 s o u r c e source source 指针指向的字符串(源字符串) 拷贝到 d e s t i n a t i o n destination destination 指针指向的字符串(目标字符串) 中。

  • 源字符串必须以 ‘\0’ 结束
  • 该函数会将 ‘\0’ 拷贝到目标空间
  • 目标空间必须足够大,以确保能存放源字符串,否则会出现越界访问的情况
  • 目标空间必须是可修改
  • 函数的返回值为指向目标空间的指针

  
  可能会有小伙伴说,字符串的拷贝哪有那么麻烦,还要用函数,不就是直接赋值吗?

int main()
{char arr1[] = "abcdef";char arr2[20] = { 0 };arr2 = arr1;return 0;
}

  这样可以吗?当然是不可以的。为什么呢?
  
  别忘了 a r r 1 arr1 arr1 a r r 2 arr2 arr2 是两个数组首元素的地址,地址是一个常量值并不是变量,就像:5 = 3;,可以把 3 赋值给 5 吗?肯定是不行的。因此字符串的赋值并没有那么简单哦。
  

2.2、函数的使用

  

#include<stdio.h>
#include<string.h>int main()
{char src[20] = "good good study";char dest[40] = { 0 };char* s = strcpy(dest, src);printf("%s\n", s);return 0;
}

  
运行结果
在这里插入图片描述
  

2.3、函数的模拟实现

  要模拟实现 s t r c p y strcpy strcpy 函数,其实思路很简单,只需将字符串的中的每个字符依次拷贝过去即可,当拷贝完源字符串的 '\0'拷贝结束
  
  我们来尝试写下代码:

char* my_strcpy(char* dest, const char* src)
{//要返回目标空间的地址char* ret = dest;//不能传入空指针assert(dest != NULL);assert(src != NULL);//拷贝'\0'之前的内容while (*src != '\0'){*dest = *src;dest++;src++;}//拷贝'\0'*dest = *src;return ret;
}

  
  上述代码基本上是没问题的,但大家发现没有,该代码有点繁琐*dest = *src;就出现了两次,有什么改进方法呢?我们来看下面的代码:
  

char* my_strcpy(char* dest, const char* src)
{char* ret = dest;assert(dest && src);while (*dest++ = *src++){;}return ret;
}

  
这里,我们重点来讲解一下while (*dest++ = *src++);语句

  • 虽然两个 后置++ 的优先级最高,但是因为是后置:先使用,后++
  • 首先执行的是两个解引用操作
  • 解引用完成后,将 * s r c src src 的值赋给 * d e s t dest dest
  • 赋值完成后进行循环的条件判断,* d e s t dest dest == ‘\0’ 则退出循环,否则执行循环
  • 最后 d s e t dset dset s r c src src 两个指针执行++,即自增
      

怎么样,这段代码是不是很巧妙
  
  

三、 s t r c n t strcnt strcnt 函数

3.1、函数功能

在这里插入图片描述

功能 s t r c a t strcat strcat 函数的功能是实现字符创的追加,将源字符串的内容追加到目标字符串的结尾。

  • 源字符串必须以 ‘\0’ 结束
  • 追加会将目标空间原来的 ‘\0’ 覆盖
  • 目标字符串结尾也要有 ‘\0’,否则不知从哪开始追加
  • 目标空间必须足够大,能容下追加源字符串之后的内容
  • 目标空间必须可修改
  • 返回的是目标空间的起始地址
  • 目标空间与源字符串不能重叠,即不能自己给自己追加

  

3.2、函数的使用

  

#include<stdio.h>
#include<string.h>
int main()
{char src[20] = "day day up!";char dest[40] = "good good study! ";char* s = strcat(dest, src);printf("%s\n", s);return 0;
}

  
运行结果
在这里插入图片描述
  

3.3、函数的模拟实现

  
   s t r c a t strcat strcat 函数怎么模拟实现呢?大家不妨想一想,追加不也是一种拷贝吗,它的实现方法与 s t r c p y strcpy strcpy 函数的实现很像,只是 s t r c p y strcpy strcpy从头开始拷贝,而 s t r c a t strcat strcat从目标字符串的末尾开始拷贝。
  
参考代码

char* mu_strcat(char* dest, const char* src)
{char* ret = dest;assert(dest && src);//找到目标空间的末尾while (*dest){dest++;}//进行追加(拷贝)while (*dest++ = *src++){;}return ret;
}

  
  

四、 s t r c m p strcmp strcmp 函数

4.1、函数功能

在这里插入图片描述

功能比较两个字符串的大小
    s t r 1 str1 str1 > s t r 2 str2 str2 ,返回 > 0 >0 >0 的数字
    s t r 1 str1 str1 = s t r 2 str2 str2 ,返回 0 0 0
    s t r 1 str1 str1 < s t r 2 str2 str2 ,返回 < 0 <0 <0 的数字
  
  
比较方式
  
   s t r c m p strcmp strcmp 函数比较字符串中对应位置上两个字符的 ASCII码 值的大小来比较两个字符串大小

   s t r c m p strcmp strcmp逐一比较两个字符串中相应位置的各个字符,从第一个字符开始比较,若相等,则比较下位字符
  直至比出大小遇到其中一个字符串的 '\0',比较结束,先遇到'\0'的字符串较小,可以理解成 ‘\0’ 的ASCII码值为 0,任何字符的 ASCII码值都比它大。
  

4.2、函数的使用

#inlcude<stdio.h>
int main()
{char str1[] = "abcd";char str2[] = "Abcd";char str3[] = "abcd";char str4[] = "bbcd";int ret1 = strcmp(str1, str2);//比较str1与str2int ret2 = strcmp(str1, str3);//比较str1与str3int ret3 = strcmp(str1, str4);//比较str1与str4printf("%d %d %d\n", ret1, ret2, ret3);return 0;
}

  
运行结果
在这里插入图片描述

  

4.3、函数的模拟实现

int my_strcmp(const char* str1, const char* str2)
{//接下来要对指针进行解引用,因此不能传入空指针assert(str1 && str2);//相同位上两个字符相等,进入循环while (*str1 == *str2){//当两个字符都为'\0'时,也进入了循环,遇到'\0'表示字符串结束//因为相等才能进入循环,判断语句因此只用写一个if (*str1 == '\0'){return 0;}str1++;str2++;}//不相等,返回两个字符的ASCII码的差值return *str1 - *str2;
}

  
  
  
  


  好啦,本期关于部分字符串函数的介绍及模拟实现就介绍到这里啦,希望本期博客能对你有所帮助,更多关于字符串的函数还请收看下一期。同时,如果有错误的地方请多多指正,让我们在C语言的学习路上一起进步!

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

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

相关文章

tkinter窗口组件Entry

from tkinter import * 创建主窗口 app Tk() 设置窗口大小为1040x2048(手机) app.geometry(“1040x2048”) 设置窗口背景为灰色 app.configure(bg“gray”) 定义一个函数text()&#xff0c;用于处理输入框内容的变化 def text(): # 获取输入框e1的内容 if e1.get() “…

JS/TS笔记学习1

周末总得学点什么吧~ 奥利给! 跑火车 递归 减速 let currentIndex 0; let speed 500; // 初始速度&#xff0c;单位是毫秒 let decrement 20; // 每次迭代速度减少的量 const cells document.querySelectorAll(.cell); function highlightCell() { cells.forEach(…

常见的锁策略,synchronized优化过程和cas过程

1. 常见的锁策略 所谓"策略",也可以理解为做法."锁策略"就是用来描述一把锁面对加锁/解锁时的做法. 1.1 乐观锁 vs 悲观锁 要区分一把锁是乐观锁还是悲观锁,就要预测当前这把锁冲突的概率高不高. 如果冲突概率高,后续要做的工作往往会更多,加锁的开销就…

使用阿里云试用Elasticsearch学习:4. 聚合——2

近似聚合 如果所有的数据都在一台机器上&#xff0c;那么生活会容易许多。 CS201 课上教的经典算法就足够应付这些问题。如果所有的数据都在一台机器上&#xff0c;那么也就不需要像 Elasticsearch 这样的分布式软件了。不过一旦我们开始分布式存储数据&#xff0c;就需要小心…

Docker 安装 RocketMQ

目录 一、新建两个配置文件 1.1 创建docker-compose.yml文件 1.2 .新建broker.conf文件 二、运行 三、可视化界面 一、新建两个配置文件 1.1 创建docker-compose.yml文件 version: 3.5 services:rmqnamesrv:image: foxiswho/rocketmq:servercontainer_name: rmqnamesrvports…

用html写一个雨的特效

<!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>雨特效</title><link rel"stylesheet" href"./style.css"> </head> <body> <div id"wrap-textu…

43岁百亿千金夫妇国外旅游高调放闪,羡煞好友马国明

萧正楠早前忙着拍摄TVB新剧《奔跑吧&#xff01;勇敢的女人们》&#xff0c;剧集煞科后他即和老婆黄翠如一起到意大利旅行。两公婆分别在IG大晒旅行靓相&#xff0c;甜蜜满泻。 这次萧正楠的旅游照都是由翠如亲自操刀拍摄&#xff0c;相中所见&#xff0c;萧正楠在广场大摆Chok…

竞赛 协同过滤电影推荐系统

文章目录 1 简介1 设计概要2 课题背景和目的3 协同过滤算法原理3.1 基于用户的协同过滤推荐算法实现原理3.1.1 步骤13.1.2 步骤23.1.3 步骤33.1.4 步骤4 4 系统实现4.1 开发环境4.2 系统功能描述4.3 系统数据流程4.3.1 用户端数据流程4.3.2 管理员端数据流程 4.4 系统功能设计 …

AI人工智能讲师大模型培训讲师叶梓 大语言模型(LLM)在科学文献摘要领域的应用

大语言模型&#xff08;LLM&#xff09;在科学文献摘要领域的应用是一个前沿且迅速发展的技术趋势。通过结合GitHub上yobibyte的Compressor项目&#xff0c;我们可以深入探讨这一技术方案的潜力和实现方式。 技术背景 随着科学研究的快速发展&#xff0c;每天都有大量的科学文…

【软考中级】软件设计师考点分布

文章目录 软考官网资格设置软考报考流程 【软件设计师】考点分布选择题考点分布案例题考点分布 软考官网 中国计算机技术职业资格网&#xff1a;https://www.ruankao.org.cn/ 官网报名平台&#xff1a;https://bm.ruankao.org.cn/sign/welcome 资格设置 计算机软件计算机网…

QT常用控件

常用控件 控件概述QWidget 核⼼属性核⼼属性概览enabledgeometrywindowTitlewindowIconwindowOpacitycursorfonttoolTipfocusPolicystyleSheet 按钮类控件Push ButtonRadio ButtionCheck Box 显⽰类控件LabelLCD NumberProgressBarCalendar Widget 输⼊类控件Line EditText Edi…

MemberPress配置和使用会员登录页面

目录 隐藏 创建会员登录页面 编辑登录页面 设计您的登录页面 链接到您的登录页面 创建会员登录页面 要创建MemberPress会员登录页面&#xff0c;您需要做的就是导航到 MemberPress > 设置 > 页面选项卡&#xff0c;然后在页面顶部附近的“MemberPress 登录页面”…

计算机网络——ARP协议

前言 本博客是博主用于复习计算机网络的博客&#xff0c;如果疏忽出现错误&#xff0c;还望各位指正。 这篇博客是在B站掌芝士zzs这个UP主的视频的总结&#xff0c;讲的非常好。 可以先去看一篇视频&#xff0c;再来参考这篇笔记&#xff08;或者说直接偷走&#xff09;。 …

微服务边车模式深度解析:赋能云原生应用的终极指南(自己搞一个简单SideCar?)

什么是SideCar? Sidecar模式定义&#xff1a; Sidecar 模式是一种常用于微服务架构中的设计模式&#xff0c;该模式允许将应用程序的核心功能与辅助功能&#xff08;如日志记录、监控、配置管理、网络通信等&#xff09;分离开来。在这种设计模式中&#xff0c;每个微服务主容…

快团团团长如何设置起购金额?新手教程免费学!

一、功能说明 部分团长有最低起送价的需求&#xff0c;这时可以通过设置起购金额来规定团员的最低起购价。 二、具体操作步骤 &#xff08;一&#xff09;如何设置起购金额&#xff1f; 在团购设置中&#xff0c;点击“更多团购设置”&#xff0c;在“其他设置”中找到“起购…

vue3使用jsQR解析二维码

1.了解jsQR jsQR是一个纯javascript脚本实现的二维码识别库&#xff0c;不仅可以在浏览器端使用&#xff0c;而且支持后端node.js环境。jsQR使用较为简单&#xff0c;有着不错的识别率。 2.效果图 3.二维码 4.下载jsqr包 npm i -d jsqr5.代码 <script setup> import …

每日一题(leetcode209):长度最小的子数组--滑动窗口

由于是区间问题&#xff0c;考虑到使用滑动窗口。 下面是我刚开始写的代码&#xff0c;虽然能通过&#xff0c;但是思维紊乱&#xff0c;循环条件应该去盯着满足情况下的条件&#xff0c;然后每次都去比较最小值&#xff0c;这样比较简洁。 class Solution { public:int minS…

卷积神经网络结构组成与解释

卷积神经网络结构组成与解释 卷积神经网络是以卷积层为主的深度网路结构&#xff0c;网络结构包括有卷积层、激活层、BN层、池化层、FC层、损失层等。卷积操作是对图像和滤波矩阵做内积&#xff08;元素相乘再求和&#xff09;的操作。 1. 卷积层 常见的卷积操作如下&#x…

比特币减半后 牛市爆发

作者&#xff1a;Arthur Hayes of Co-Founder of 100x 编译&#xff1a;Qin jin of ccvalue (以下内容仅代表作者个人观点&#xff0c;不应作为投资决策依据&#xff0c;也不应被视为参与投资交易的建议或意见&#xff09;。 Ping PingPing&#xff0c;我的手机发出的声音&…

Servlet(一)

文章目录 1.Servlet整体框架2.Servlet快速入门1.创建项目配置基本环境2.添加jar包1.在WEB-INF下创建目录lib&#xff0c;添加文件2.添加到项目中3.配置代码提示 3.src下创建文件4.实现Servlet接口5.在web.xml配置HelloServlet6.通过浏览器访问HelloServlet 3.浏览器访问Servlet…