在PHP编程中,在遍历数组的时候经常需要先计算数组的长度作为循环结束的判断条件,而在PHP里面对数组的操作是很频繁的,因此count也算是一个常用函数,下面研究一下count函数的具体实现。
我在github上有对PHP源码更详细的注解。感兴趣的可以围观一下,给个star。PHP5.4源码注解。可以通过commit记录查看已添加的注解。
count
int <span style="color: #008080;">count</span> ( <span style="color: #0000ff;">mixed</span> <span style="color: #800080;">$array_or_countable</span> [, int <span style="color: #800080;">$mode</span> = COUNT_NORMAL ] )
count函数计算数组或者对象里面的所有元素个数。
对于对象来说,如果你安装了SPL扩展,可以通过实现Countable接口来调用count函数。Countable接口有且仅有一个方法Countable::count(),该方法的返回count()函数的返回值。
参数说明
mode
如果参数mode设为COUNT_RECURSIVE(或1),count()会递归地计算该数组。在计算多维数组的时候特别有用。
如果第一个参数不是数组或者实现Countable接口的对象,count函数将返回1。
注意:count函数可以检测递归避免无限循环,但会在遇到无限递归或得到比期望值大的时候返回E_WARNING提示。
运行示例
普通应用
<span style="color: #800080;">$arr1</span> = <span style="color: #0000ff;">array</span>(1, 2, 3, 4, 5<span style="color: #000000;">); </span><span style="color: #800080;">$val1</span> = <span style="color: #008080;">count</span>(<span style="color: #800080;">$arr1</span>); <span style="color: #008000;">//</span><span style="color: #008000;"> 5</span>
多维数组
<span style="color: #800080;">$arr2</span> = <span style="color: #0000ff;">array</span>('apple', 'banana', <span style="color: #0000ff;">array</span>('cat', 'camel'), 'dog'<span style="color: #000000;">); </span><span style="color: #800080;">$val2_1</span> = <span style="color: #008080;">count</span>(<span style="color: #800080;">$arr2</span>); <span style="color: #008000;">//</span><span style="color: #008000;"> 4</span> <span style="color: #800080;">$val2_2</span> = <span style="color: #008080;">count</span>(<span style="color: #800080;">$arr2</span>, 1); <span style="color: #008000;">//</span><span style="color: #008000;"> 6</span>
数字和字符串
<span style="color: #800080;">$str</span> = "hello world"<span style="color: #000000;">; </span><span style="color: #800080;">$int_val</span> = 1<span style="color: #000000;">; </span><span style="color: #800080;">$val3</span> = <span style="color: #008080;">count</span>(<span style="color: #800080;">$str</span>); <span style="color: #008000;">//</span><span style="color: #008000;"> 1</span> <span style="color: #800080;">$val4</span> = <span style="color: #008080;">count</span>(<span style="color: #800080;">$int_val</span>); <span style="color: #008000;">//</span><span style="color: #008000;"> 1</span>
普通对象
<span style="color: #0000ff;">class</span><span style="color: #000000;"> User { </span><span style="color: #0000ff;">private</span> <span style="color: #800080;">$name</span><span style="color: #000000;">; </span><span style="color: #0000ff;">private</span> <span style="color: #800080;">$address</span><span style="color: #000000;">; } </span><span style="color: #800080;">$user</span> = <span style="color: #0000ff;">new</span><span style="color: #000000;"> User(); </span><span style="color: #800080;">$val5</span> = <span style="color: #008080;">count</span>(<span style="color: #800080;">$user</span>); <span style="color: #008000;">//</span><span style="color: #008000;"> 1</span> <span style="color: #800080;">$val6</span> = <span style="color: #008080;">count</span>((<span style="color: #0000ff;">array</span>) <span style="color: #800080;">$user</span>); <span style="color: #008000;">//</span><span style="color: #008000;"> 2</span>
array-like对象
<span style="color: #0000ff;">class</span> User <span style="color: #0000ff;">extends</span><span style="color: #000000;"> ArrayObject { </span><span style="color: #0000ff;">private</span> <span style="color: #800080;">$name</span><span style="color: #000000;">; </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">function</span><span style="color: #000000;"> __construct() { </span><span style="color: #800080;">$this</span>->name = 'hhq'<span style="color: #000000;">; } </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">function</span><span style="color: #000000;"> getName() { </span><span style="color: #0000ff;">return</span> <span style="color: #800080;">$this</span>-><span style="color: #000000;">name; } </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">function</span> <span style="color: #008080;">count</span><span style="color: #000000;">() { </span><span style="color: #0000ff;">return</span> 2<span style="color: #000000;">; } } </span><span style="color: #800080;">$user2</span> = <span style="color: #0000ff;">new</span><span style="color: #000000;"> User(); </span><span style="color: #800080;">$val7</span> = <span style="color: #008080;">count</span>(<span style="color: #800080;">$user2</span>); <span style="color: #008000;">//</span><span style="color: #008000;"> 2</span>
实现Countable接口对象
<span style="color: #0000ff;">class</span> User <span style="color: #0000ff;">implements</span><span style="color: #000000;"> Countable { </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">function</span> <span style="color: #008080;">count</span><span style="color: #000000;">() { </span><span style="color: #0000ff;">return</span> 3<span style="color: #000000;">; } } </span><span style="color: #800080;">$user3</span> = <span style="color: #0000ff;">new</span><span style="color: #000000;"> User(); </span><span style="color: #800080;">$val8</span> = <span style="color: #008080;">count</span>(<span style="color: #800080;">$user3</span>); <span style="color: #008000;">//</span><span style="color: #008000;"> 3</span>
运行步骤
进入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,最终得到了想要的答案。不懂就要问,哈哈。
原创文章,文笔有限,才疏学浅,文中若有不正之处,万望告知。
如果本文对你有帮助,请点下推荐吧,谢谢^_^
最后再安利一下,我在github有对PHP源码更详细的注解。感兴趣的可以围观一下,给个star。PHP5.4源码注解。可以通过commit记录查看已添加的注解。
更多源码文章,欢迎访问个人主页继续查看:hoohack

핫 AI 도구

Undresser.AI Undress
사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover
사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool
무료로 이미지를 벗다

Clothoff.io
AI 옷 제거제

AI Hentai Generator
AI Hentai를 무료로 생성하십시오.

인기 기사

뜨거운 도구

드림위버 CS6
시각적 웹 개발 도구

맨티스BT
Mantis는 제품 결함 추적을 돕기 위해 설계된 배포하기 쉬운 웹 기반 결함 추적 도구입니다. PHP, MySQL 및 웹 서버가 필요합니다. 데모 및 호스팅 서비스를 확인해 보세요.

DVWA
DVWA(Damn Vulnerable Web App)는 매우 취약한 PHP/MySQL 웹 애플리케이션입니다. 주요 목표는 보안 전문가가 법적 환경에서 자신의 기술과 도구를 테스트하고, 웹 개발자가 웹 응용 프로그램 보안 프로세스를 더 잘 이해할 수 있도록 돕고, 교사/학생이 교실 환경 웹 응용 프로그램에서 가르치고 배울 수 있도록 돕는 것입니다. 보안. DVWA의 목표는 다양한 난이도의 간단하고 간단한 인터페이스를 통해 가장 일반적인 웹 취약점 중 일부를 연습하는 것입니다. 이 소프트웨어는

MinGW - Windows용 미니멀리스트 GNU
이 프로젝트는 osdn.net/projects/mingw로 마이그레이션되는 중입니다. 계속해서 그곳에서 우리를 팔로우할 수 있습니다. MinGW: GCC(GNU Compiler Collection)의 기본 Windows 포트로, 기본 Windows 애플리케이션을 구축하기 위한 무료 배포 가능 가져오기 라이브러리 및 헤더 파일로 C99 기능을 지원하는 MSVC 런타임에 대한 확장이 포함되어 있습니다. 모든 MinGW 소프트웨어는 64비트 Windows 플랫폼에서 실행될 수 있습니다.

SecList
SecLists는 최고의 보안 테스터의 동반자입니다. 보안 평가 시 자주 사용되는 다양한 유형의 목록을 한 곳에 모아 놓은 것입니다. SecLists는 보안 테스터에게 필요할 수 있는 모든 목록을 편리하게 제공하여 보안 테스트를 더욱 효율적이고 생산적으로 만드는 데 도움이 됩니다. 목록 유형에는 사용자 이름, 비밀번호, URL, 퍼징 페이로드, 민감한 데이터 패턴, 웹 셸 등이 포함됩니다. 테스터는 이 저장소를 새로운 테스트 시스템으로 간단히 가져올 수 있으며 필요한 모든 유형의 목록에 액세스할 수 있습니다.
