在PHP编程中,在遍历数组的时候经常需要先计算数组的长度作为循环结束的判断条件,而在PHP里面对数组的操作是很频繁的,因此count也算是一个常用函数,下面研究一下count函数的具体实现。
我在github上有对PHP源码更详细的注解。感兴趣的可以围观一下,给个star。PHP5.4源码注解。可以通过commit记录查看已添加的注解。
count
int count ( mixed $array_or_countable [, int $mode = COUNT_NORMAL ] )
count函数计算数组或者对象里面的所有元素个数。
对于对象来说,如果你安装了SPL扩展,可以通过实现Countable接口来调用count函数。Countable接口有且仅有一个方法Countable::count(),该方法的返回count()函数的返回值。
参数说明
mode
如果参数mode设为COUNT_RECURSIVE(或1),count()会递归地计算该数组。在计算多维数组的时候特别有用。
如果第一个参数不是数组或者实现Countable接口的对象,count函数将返回1。
注意:count函数可以检测递归避免无限循环,但会在遇到无限递归或得到比期望值大的时候返回E_WARNING提示。
运行示例
普通应用
$arr1 = array(1, 2, 3, 4, 5); $val1 = count($arr1); // 5
多维数组
$arr2 = array('apple', 'banana', array('cat', 'camel'), 'dog'); $val2_1 = count($arr2); // 4 $val2_2 = count($arr2, 1); // 6
数字和字符串
$str = "hello world"; $int_val = 1; $val3 = count($str); // 1 $val4 = count($int_val); // 1
普通对象
class User { private $name; private $address; } $user = new User(); $val5 = count($user); // 1 $val6 = count((array) $user); // 2
array-like对象
class User extends ArrayObject { private $name; public function __construct() { $this->name = 'hhq'; } public function getName() { return $this->name; } public function count() { return 2; } } $user2 = new User(); $val7 = count($user2); // 2
实现Countable接口对象
class User implements Countable { public function count() { return 3; } } $user3 = new User(); $val8 = count($user3); // 3
运行步骤
进入switch语句检测参数类型
如果是NULL,直接返回0
如果是数组,调用php_count_recursive函数机选数组元素个数
如果是对象,先检查是否为数组对象(array-like object),如果是,则计算数组对象的数量
否则,如果对象实现了Countable接口,则调用Countable的count方法
最后,其他类型比如整型数组或字符串,都返回1。
源码解读
如果是普通数组,count函数会调用php_count_recursive函数实现其功能的运行步骤如下:
如果当前hash Bucket被递归访问的次数大于1,说明重复递归,染回E_WARNING错误
否则计算当前数组层数的数组元素个数
如果有递归参数选项,则继续递归访问
如果参数是对象类型,实现时会先判断handler是否被定义。而handler是PHP内核中对象的结构体,其中包含有count_elements字段,实际上是一个函数。如果某个对象表现得想数组一样,即通常说的array-like object,那么就会执行count_elements函数。具体实现是类继承PHP的ArrayObject,并在类里面实现count函数,具体调用的就是count函数,如果类没有实现count函数,则count返回0,否则返回对象的count函数的返回值。
如果是其他的数据类型
1、字符串
2、数字
3、对象分支中两个if判断都为false的情况,即没有继承ArrayObject且没有实现Countable接口。
这些类型通通返回1。
需要注意的是,如果需要计算的是对象的属性数量,可以先将对象转换成数组,然后调用count函数。如:
$count_value = count((array) $user);
小结
阅读count函数的源码过程中,在其中一步卡住了,就是if (Z_OBJ_HT_P(array)->count_elements)这一步,因为始终无法写出进入这个分支的demo,在网上搜索了很多资料也未果,因此请教了TIPI的reeze,最终得到了想要的答案。不懂就要问,哈哈。
本文向大家介绍深入理解PHP 数组之count 函数,包括了深入理解PHP 数组之count 函数的使用技巧和注意事项,需要的朋友参考一下 count() PHP count() 函数用于计算数组中的单元数目或对象中的属性个数,返回数组的单元个数或对象中的属性个数。 语法: int count( mixed var [, int mode] )如果 var 是非数组的普通变量,则返回 1 ,对于不
本文向大家介绍深入理解PHP中的empty和isset函数,包括了深入理解PHP中的empty和isset函数的使用技巧和注意事项,需要的朋友参考一下 近日被问到PHP中empty和isset函数时怎么判断变量的,刚开始我是一脸懵逼的,因为我自己也只是一知半解,为了弄懂其真正的原理,赶紧翻开源码研究研究。经过分析可发现两个函数调用的都是同一个函数,因此本文将对两个函数一起分析。 我在github有
本文向大家介绍PHP的imageTtfText()函数深入详解,包括了PHP的imageTtfText()函数深入详解的使用技巧和注意事项,需要的朋友参考一下 图片处理,是许多程序的功能之一;而文字渲染则是绘图的基本组成部分。PHP通过很多扩展库来支持图片的处理,最常用的还是GD库,通过一系列imagexxx()函数来提供绘图功能。本文专注于非常细小的一点:绘制文字。熟悉Win32的人都知道,Te
本文向大家介绍深入理解PHP中的Streams工具,包括了深入理解PHP中的Streams工具的使用技巧和注意事项,需要的朋友参考一下 Streams 是PHP提供的一个强有力的工具,我们常常在不经意会使用到它,如果善加利用将大大提高PHP的生产力。 驾驭Streams的强大力量后,应用程序将提升到一个新的高度。 下面是PHP手册中对Streams的一段描述: Streams 是在PHP
TIPI 项目是一个自发项目,项目主要关注PHP的内部实现,以及PHP相关的方方面面,该项目包括《深入理解PHP内核》这本书,以及一些相关的PHP项目,例如一些 PHP 扩展及研究项目。
本文向大家介绍深入理解JS addLoadEvent函数,包括了深入理解JS addLoadEvent函数的使用技巧和注意事项,需要的朋友参考一下 首先是addLoadEvent函数的代码清单: 理解起来就是: 1.把现有的window.onload事件处理函数的值存入变量 oldonload; 2.如果在这个处理函数上还没有绑定任何函数,就像平常那样把新函数添加给它; 3.如果在这个处理函数上已
本文向大家介绍深入理解vue Render函数,包括了深入理解vue Render函数的使用技巧和注意事项,需要的朋友参考一下 最近在学习vue,正好今日留个笔记,我自己还在摸索学习中,现整理出来以作记录。 会使用基本的Render函数后,就会想,这怎么用 v-for/v-if/v-model;我写个vue Render函数进阶 首先是v-if 的转化使用全局组件的v 值决定组件渲染的状态,对实例
让我们深入研究一下箭头函数。 箭头函数不仅仅是编写简洁代码的“捷径”。它还具有非常特殊且有用的特性。 JavaScript 充满了我们需要编写在其他地方执行的小函数的情况。 例如: arr.forEach(func) —— forEach 对每个数组元素都执行 func。 setTimeout(func) —— func 由内建调度器执行。 ……还有更多。 JavaScript 的精髓在于创建一个