一、strlen()的工作原理

二、模拟实现strlen的三种方法

  1. 计数器方法
  2. 指针-指针
  3. 递归的方法

三、库函数实现strlen的思路

四、库函数的strlen同上面模拟实现strlen的区别

一、strlen工作原理

strlen函数工作原理:是计算字符串str的长度,直到空字符串结束,但不包含空字符串。(即该长度算至/0结束,但不包含/0)

通过以下代码能有一个直观的感受:

int main(){
char arr[]="abcdef";
int len=strlen(a);//"abcdef"即 'a' 'b' 'c' 'd' 'e' 'f' '\0'
//计算的是'\0'之前的元素个数
printf("%d\n",len); // 输出结果为6
return 0; }

以上则为strlen的工作原理

即strlen计算的是字符串内'\0'之前的元素个数。(不包含'\0')

二、模拟实现strlen的三种方法

  1. 计数器方法
  2. 指针-指针
  3. 递归的方法

我们通过上述得知strlen的工作原理,那么我们接下来根据其工作原理实现strlen.

共有以下三种方式去实现:

ps:只有sizeof(数组名)&(数组名),此时代表的是整个数组的地址,其余时候

数组名 仅仅代表首元素地址。在本文中,数组名 只代表首元素地址。

int my_strlen(){}    //要实现的功能,如下三种方式
int main(){
char arr[]="abcdef";
int len=my_strlen(a);
printf("%d\n",len);
return 0; }

1.计数器方式(创建了临时变量)

思路如下:将arr传递到my_strlen中,我们创建一个变量 count ,当 *arr!=0 时则 count +1, arr也+1 ,此时 *arr='b' ,arr和count再同时+1, 直到 *a=='\0'时,

我们返回 count 的值,此时的 count 的值为元素的个数。

图解如下:

算法如下:

int my_strlen(const char *str){
int count=0;
while(*str!='\0'){ //当为'\0'时,即while(0),条件为假,退出循环
count++;
str++;
}
return count;
}

2.指针-指针(创建了临时变量)

思路如下:创建指针p指向arr,当p!='\0'时,p加上1,直到p='\0',用

指针p-指针arr,此时的指针p代表的是'\0'的地址,指针arr代表的是首

元素地址,两者相减得到的值为两地址之间的元素个数。

图解如下:

当p走到'\0'的地址时, p-arr=元素的个数

实现代码如下:

 int my_strlen(const char*str){
const char*p=str; while(*p){
p++;
}
return p-str;
}

3.递归(不创建临时变量)

思路如下:通过递归的思想。

图解吐下:

ps:递归的基本思想是把一个大型复杂的问题层层转化为一个与原问题相似的规

模较小的问题来求解。

算法如下:

int my_strlen(const char*str){
if(*str = = '\0'){
return 0;
}
else
return 1+my_strlen(str+1)
}

三、参考库函数如何实现strlen

对比自身用指针实现strlen,代码更为简洁,将 while(*p){p++;} 精简为

while(*p++);

图解如下:

ps:如何查看库函数实现strlen。可看链接:如何查看库函数实现的某些函数(strlen,strcmp,strcpy等)

四、库函数的strlen同上面模拟实现strlen的区别

可通过如下代码进行区别:

#include<string.h>
#include<stdio.h>
int main(){
if(strlen("abc")-strlen("abcdef")>0)
{printf("hehe\n");
}
else
{printf("haha\n");
}
}

很显然,我们应该是打印 haha,但结果是hehe,为什么打印的是hehe

因为在strlen()的函数声明如下:size_t strlen(const char*str)

定义的类型是size_t,那么size_t又是什么类型

通过如下操作可知:

可知size_t是无符号整型,于是:无符号整型-无符号整型=无符号整型

于是打印的是hehe

我们模拟实现函数strlen返回的是int,于是当我们

调用my_strlen时,打印的便是haha,效果如下

模拟实现strlen的三种方法的更多相关文章

  1. JS模拟实现封装的三种方法

      前  言  继承是使用一个子类继承另一个父类,那么子类可以自动拥有父类中的所有属性和方法,这个过程叫做继承!  JS中有很多实现继承的方法,今天我给大家介绍其中的三种吧. 1.在 Object类上 ...

  2. 三种方法教你如何用PHP模拟post提交数据

    php模拟post传值在日常的工作中用到的不是很多,但是在某些特定的场合还是经常用到的. 下面,我整理了三种php模拟post传值的方法,file_get_contents.curl和socket. ...

  3. Chrome模拟手机浏览器(iOS/Android)的三种方法,亲测无误!

    大网站都有推出自己的手机访问版本页面,不管是新闻类还是视频网站,我们在电脑是无法直接访问到手机网站的,比如我经常访问一个3g.qq.com这个手机站点,如果在电脑上直接打开它,则会跳转到其它页面,一般 ...

  4. 用Fiddler可以设置浏览器的UA 和 手动 --Chrome模拟手机浏览器(iOS/Android)的三种方法,亲测无误!

    附加以一种软件的方法是:用Fiddler可以设置浏览器的UA 以下3种方法是手动的 通过伪装User-Agent,将浏览器模拟成Android设备. 第一种方法:新建Chrome快捷方式 右击桌面上的 ...

  5. Javascript定义类(class)的三种方法

    将近20年前,Javascript诞生的时候,只是一种简单的网页脚本语言.如果你忘了填写用户名,它就跳出一个警告. 如今,它变得几乎无所不能,从前端到后端,有着各种匪夷所思的用途.程序员用它完成越来越 ...

  6. javascript生成对象的三种方法

    /** js生成对象的三种方法*/ // 1.通过new Object,然后添加属性 示例如下: var people1 = new Object(); people1.name = 'xiaohai ...

  7. php生成随机数的三种方法

    php生成随机数的三种方法 如何用php生成1-10之间的不重复随机数? 例1,使用shuffle函数生成随机数. <?php$arr=range(1,10);shuffle($arr);for ...

  8. [转]Javascript定义类的三种方法

    作者: 阮一峰 原文地址:http://www.ruanyifeng.com/blog/2012/07/three_ways_to_define_a_javascript_class.html 将近2 ...

  9. php发送post请求的三种方法示例

    本文分享下php发送post请求的三种方法与示例代码,分别使用curl.file_get_content.fsocket来实现post提交数据,大家做个参考. php发送post请求的三种方法,分别使 ...

  10. java 获取随机数的三种方法

    方法1(数据类型)(最小值+Math.random()*(最大值-最小值+1))例:(int)(1+Math.random()*(10-1+1))从1到10的int型随数 方法2获得随机数for (i ...

随机推荐

  1. Go语言核心36讲51

    你好,我是郝林,今天我们继续分享程序性能分析基础的内容. 在上一篇文章中,我们围绕着"怎样让程序对CPU概要信息进行采样"这一问题进行了探讨,今天,我们再来一起看看它的拓展问题. ...

  2. 【DL论文精读笔记】Image Segmentation Using Deep Learning: A Survey 图像分割综述

    深度学习图像分割综述 Image Segmentation Using Deep Learning: A Survey 原文连接:https://arxiv.org/pdf/2001.05566.pd ...

  3. 我的Vue之旅 11 Vuex 实现购物车

    Vue CartView.vue script 数组的filter函数需要return显式返回布尔值,该方法得到一个新数组. 使用Vuex store的modules方式,注意读取状态的方式 this ...

  4. Day24:static关键字

    static static关键字是静态的意思,可以修饰成员方法.属性. static修饰的特点: 被类的所有对象共享 可以通过类名调用,也可以通过对象名调用:推荐使用类名调用! public clas ...

  5. 关于python3调用matplotlib中文乱码问题

    问题描述 我用来绘制柱形图,横坐标上面的数据, 但是网上大部分说的都是更改横纵坐标标签的乱码问题,而不是横坐标数据乱码问题 解决办法 更改横纵坐标上标签的中文不乱码 import matplotlib ...

  6. 我开发的开源项目,让.NET7中的EFCore更轻松地使用强类型Id

    在领域驱动设计(DDD)中,有一个非常重要的概念:"强类型Id".使用强类型Id来做标识属性的类型会比用int.Guid等通用类型能带来更多的好处.比如有一个根据根据Id删除用户的 ...

  7. Zabbix技术分享——监控windows进程资源使用情况

    监控系统进程资源的使用情况是IT运维的常规操作.在实际工作中,运维人员有可能遇到可以使用Zabbix Agent监控linux系统进程,却无法监控windows系统进程的情况.这是因为Zabbix A ...

  8. Java单例模式的最佳实践?

    "读过书,--我便考你一考.茴香豆的茴字,怎样写的?"--鲁迅<孔乙己> 0x00 大纲 目录 0x00 大纲 0x01 前言 0x02 单例的正确性 new关键字 c ...

  9. PyTorch复现LeNet-5手写识别学习笔记

    用PyTorch搭建LeNet-5手写识别 首先申明,这篇博客用于记录本人看完LeNet-5论文,并对其中的算法进行复现的记录,可以看成是学习笔记 这里只介绍复现的工作,如果想了解更多有关网络的细节, ...

  10. BeanShell 后置处理器/前置处理器实现urldecode 解码

    1.使用正则/Json提取器提取需要解码的值 2.在提取的接口中添加后置处理器或在下个调用接口中添加前置处理器 3.编码实现 String token = vars.get("access_ ...