摘要:源碼解讀如果是普通數(shù)組,函數(shù)會(huì)調(diào)用函數(shù)實(shí)現(xiàn)其功能的運(yùn)行步驟如下如果當(dāng)前被遞歸訪(fǎng)問(wèn)的次數(shù)大于,說(shuō)明重復(fù)遞歸,染回錯(cuò)誤否則計(jì)算當(dāng)前數(shù)組層數(shù)的數(shù)組元素個(gè)數(shù)如果有遞歸參數(shù)選項(xiàng),則繼續(xù)遞歸訪(fǎng)問(wèn)如果參數(shù)是對(duì)象類(lèi)型,實(shí)現(xiàn)時(shí)會(huì)先判斷是否被定義。
在PHP編程中,在遍歷數(shù)組的時(shí)候經(jīng)常需要先計(jì)算數(shù)組的長(zhǎng)度作為循環(huán)結(jié)束的判斷條件,而在PHP里面對(duì)數(shù)組的操作是很頻繁的,因此count也算是一個(gè)常用函數(shù),下面研究一下count函數(shù)的具體實(shí)現(xiàn)。
我在github有對(duì)PHP源碼更詳細(xì)的注解。感興趣的可以圍觀(guān)一下,給個(gè)star。PHP5.4源碼注解??梢酝ㄟ^(guò)commit記錄查看已添加的注解。
countint count ( mixed $array_or_countable [, int $mode = COUNT_NORMAL ] )
count函數(shù)計(jì)算數(shù)組或者對(duì)象里面的所有元素個(gè)數(shù)。
對(duì)于對(duì)象來(lái)說(shuō),如果你安裝了SPL擴(kuò)展,可以通過(guò)實(shí)現(xiàn)Countable接口來(lái)調(diào)用count函數(shù)。Countable接口有且僅有一個(gè)方法Countable::count(),該方法的返回count()函數(shù)的返回值。
參數(shù)說(shuō)明mode
如果參數(shù)mode設(shè)為COUNT_RECURSIVE(或1),count()會(huì)遞歸地計(jì)算該數(shù)組。在計(jì)算多維數(shù)組的時(shí)候特別有用。
如果第一個(gè)參數(shù)不是數(shù)組或者實(shí)現(xiàn)Countable接口的對(duì)象,count函數(shù)將返回1。
注意:count函數(shù)可以檢測(cè)遞歸避免無(wú)限循環(huán),但會(huì)在遇到無(wú)限遞歸或得到比期望值大的時(shí)候返回E_WARNING提示。
運(yùn)行示例 普通應(yīng)用$arr1 = array(1, 2, 3, 4, 5); $val1 = count($arr1); // 5多維數(shù)組
$arr2 = array("apple", "banana", array("cat", "camel"), "dog"); $val2_1 = count($arr2); // 4 $val2_2 = count($arr2, 1); // 6數(shù)字和字符串
$str = "hello world"; $int_val = 1; $val3 = count($str); // 1 $val4 = count($int_val); // 1普通對(duì)象
class User { private $name; private $address; } $user = new User(); $val5 = count($user); // 1 $val6 = count((array) $user); // 2array-like對(duì)象
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實(shí)現(xiàn)Countable接口對(duì)象
class User implements Countable { public function count() { return 3; } } $user3 = new User(); $val8 = count($user3); // 3運(yùn)行步驟
源碼解讀進(jìn)入switch語(yǔ)句檢測(cè)參數(shù)類(lèi)型
如果是NULL,直接返回0
如果是數(shù)組,調(diào)用php_count_recursive函數(shù)機(jī)選數(shù)組元素個(gè)數(shù)
如果是對(duì)象,先檢查是否為數(shù)組對(duì)象(array-like object),如果是,則計(jì)算數(shù)組對(duì)象的數(shù)量
否則,如果對(duì)象實(shí)現(xiàn)了Countable接口,則調(diào)用Countable的count方法
最后,其他類(lèi)型比如整型數(shù)組或字符串,都返回1。
如果是普通數(shù)組,count函數(shù)會(huì)調(diào)用php_count_recursive函數(shù)實(shí)現(xiàn)其功能的運(yùn)行步驟如下:
如果當(dāng)前hash Bucket被遞歸訪(fǎng)問(wèn)的次數(shù)大于1,說(shuō)明重復(fù)遞歸,染回E_WARNING錯(cuò)誤
否則計(jì)算當(dāng)前數(shù)組層數(shù)的數(shù)組元素個(gè)數(shù)
如果有遞歸參數(shù)選項(xiàng),則繼續(xù)遞歸訪(fǎng)問(wèn)
如果參數(shù)是對(duì)象類(lèi)型,實(shí)現(xiàn)時(shí)會(huì)先判斷handler是否被定義。而handler是PHP內(nèi)核中對(duì)象的結(jié)構(gòu)體,其中包含有count_elements字段,實(shí)際上是一個(gè)函數(shù)。如果某個(gè)對(duì)象表現(xiàn)得想數(shù)組一樣,即通常說(shuō)的array-like object,那么就會(huì)執(zhí)行count_elements函數(shù)。具體實(shí)現(xiàn)是類(lèi)繼承PHP的ArrayObject,并在類(lèi)里面實(shí)現(xiàn)count函數(shù),具體調(diào)用的就是count函數(shù),如果類(lèi)沒(méi)有實(shí)現(xiàn)count函數(shù),則count返回0,否則返回對(duì)象的count函數(shù)的返回值。
如果是其他的數(shù)據(jù)類(lèi)型
1、字符串
2、數(shù)字
3、對(duì)象分支中兩個(gè)if判斷都為false的情況,即沒(méi)有繼承ArrayObject且沒(méi)有實(shí)現(xiàn)Countable接口。
這些類(lèi)型通通返回1。
需要注意的是,如果需要計(jì)算的是對(duì)象的屬性數(shù)量,可以先將對(duì)象轉(zhuǎn)換成數(shù)組,然后調(diào)用count函數(shù)。如:
$count_value = count((array) $user);
閱讀count函數(shù)的源碼過(guò)程中,在其中一步卡住了,就是if (Z_OBJ_HT_P(array)->count_elements)這一步,因?yàn)槭冀K無(wú)法寫(xiě)出進(jìn)入這個(gè)分支的demo,在網(wǎng)上搜索了很多資料也未果,因此請(qǐng)教了TIPI的reeze,最終得到了想要的答案。不懂就要問(wèn),哈哈。
原創(chuàng)文章,文筆有限,才疏學(xué)淺,文中若有不正之處,萬(wàn)望告知。
如果本文對(duì)你有幫助,請(qǐng)點(diǎn)下推薦吧,謝謝^_^
最后再安利一下,我在github有對(duì)PHP源碼更詳細(xì)的注解。感興趣的可以圍觀(guān)一下,給個(gè)star。PHP5.4源碼注解??梢酝ㄟ^(guò)commit記錄查看已添加的注解。
更多源碼文章,歡迎訪(fǎng)問(wèn)個(gè)人主頁(yè)繼續(xù)查看:hoohack
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.hztianpu.com/yun/21633.html
摘要:和函數(shù)是用在取出數(shù)組的一段切片,還有用新的切片替換原刪除切片位置的功能。類(lèi)似中的和方法。我在有對(duì)源碼更詳細(xì)的注解。返回?cái)?shù)組中指定下標(biāo)和長(zhǎng)度的子數(shù)組切片。參數(shù)說(shuō)明設(shè)第一個(gè)參數(shù)數(shù)組的長(zhǎng)度為。如果處理后還是小于,則直接返回。 array_slice和array_splice函數(shù)是用在取出數(shù)組的一段切片,array_splice還有用新的切片替換原刪除切片位置的功能。類(lèi)似javascript中...
摘要:上次講到是如何解析大整數(shù)的,一筆帶過(guò)了的處理,再詳細(xì)閱讀該函數(shù)的源碼,以下是小分析??偨Y(jié)閱讀完這個(gè)函數(shù)的源碼,學(xué)習(xí)到的是浮動(dòng)數(shù)與字符串的互相轉(zhuǎn)換的實(shí)現(xiàn)細(xì)節(jié),字符串與浮點(diǎn)數(shù)之間的關(guān)系較復(fù)雜,之后還要繼續(xù)學(xué)習(xí)。 上次講到PHP是如何解析大整數(shù)的,一筆帶過(guò)了number_format的處理,再詳細(xì)閱讀該函數(shù)的源碼,以下是小分析。 函數(shù)原型 string number_format ( flo...
摘要:下面會(huì)從淺到深,淡淡在閱讀源碼過(guò)程中自己的理解。分拆子頁(yè)面后,每一個(gè)子頁(yè)面對(duì)應(yīng)一個(gè)文件。總結(jié)上面就是最早版本的源碼,很簡(jiǎn)潔的使用了等其目的也很簡(jiǎn)單簡(jiǎn)化相關(guān)生態(tài)的繁瑣邏輯參考源碼地址 ??dva的思想還是很不錯(cuò)的,大大提升了開(kāi)發(fā)效率,dva集成了Redux以及Redux的中間件Redux-saga,以及React-router等等。得益于Redux的狀態(tài)管理,以及Redux-saga中...
摘要:下面會(huì)從淺到深,淡淡在閱讀源碼過(guò)程中自己的理解。分拆子頁(yè)面后,每一個(gè)子頁(yè)面對(duì)應(yīng)一個(gè)文件。總結(jié)上面就是最早版本的源碼,很簡(jiǎn)潔的使用了等其目的也很簡(jiǎn)單簡(jiǎn)化相關(guān)生態(tài)的繁瑣邏輯參考源碼地址 ??dva的思想還是很不錯(cuò)的,大大提升了開(kāi)發(fā)效率,dva集成了Redux以及Redux的中間件Redux-saga,以及React-router等等。得益于Redux的狀態(tài)管理,以及Redux-saga中...
摘要:下面會(huì)從淺到深,淡淡在閱讀源碼過(guò)程中自己的理解。分拆子頁(yè)面后,每一個(gè)子頁(yè)面對(duì)應(yīng)一個(gè)文件??偨Y(jié)上面就是最早版本的源碼,很簡(jiǎn)潔的使用了等其目的也很簡(jiǎn)單簡(jiǎn)化相關(guān)生態(tài)的繁瑣邏輯參考源碼地址 ??dva的思想還是很不錯(cuò)的,大大提升了開(kāi)發(fā)效率,dva集成了Redux以及Redux的中間件Redux-saga,以及React-router等等。得益于Redux的狀態(tài)管理,以及Redux-saga中...
閱讀 1627·2021-11-24 09:39
閱讀 1183·2021-11-22 15:11
閱讀 2300·2021-11-19 11:35
閱讀 1687·2021-09-13 10:37
閱讀 2635·2021-09-03 10:47
閱讀 2240·2021-08-30 09:47
閱讀 1703·2021-08-20 09:39
閱讀 3012·2019-08-30 14:13