摘要:源碼解讀如果是普通數組,函數會調用函數實現其功能的運行步驟如下如果當前被遞歸訪問的次數大于,說明重復遞歸,染回錯誤否則計算當前數組層數的數組元素個數如果有遞歸參數選項,則繼續遞歸訪問如果參數是對象類型,實現時會先判斷是否被定義。
在PHP編程中,在遍歷數組的時候經常需要先計算數組的長度作為循環結束的判斷條件,而在PHP里面對數組的操作是很頻繁的,因此count也算是一個常用函數,下面研究一下count函數的具體實現。
我在github有對PHP源碼更詳細的注解。感興趣的可以圍觀一下,給個star。PHP5.4源碼注解。可以通過commit記錄查看已添加的注解。
countint 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); // 2array-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,最終得到了想要的答案。不懂就要問,哈哈。
原創文章,文筆有限,才疏學淺,文中若有不正之處,萬望告知。
如果本文對你有幫助,請點下推薦吧,謝謝^_^
最后再安利一下,我在github有對PHP源碼更詳細的注解。感興趣的可以圍觀一下,給個star。PHP5.4源碼注解。可以通過commit記錄查看已添加的注解。
更多源碼文章,歡迎訪問個人主頁繼續查看:hoohack
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/21633.html
摘要:和函數是用在取出數組的一段切片,還有用新的切片替換原刪除切片位置的功能。類似中的和方法。我在有對源碼更詳細的注解。返回數組中指定下標和長度的子數組切片。參數說明設第一個參數數組的長度為。如果處理后還是小于,則直接返回。 array_slice和array_splice函數是用在取出數組的一段切片,array_splice還有用新的切片替換原刪除切片位置的功能。類似javascript中...
摘要:上次講到是如何解析大整數的,一筆帶過了的處理,再詳細閱讀該函數的源碼,以下是小分析。總結閱讀完這個函數的源碼,學習到的是浮動數與字符串的互相轉換的實現細節,字符串與浮點數之間的關系較復雜,之后還要繼續學習。 上次講到PHP是如何解析大整數的,一筆帶過了number_format的處理,再詳細閱讀該函數的源碼,以下是小分析。 函數原型 string number_format ( flo...
摘要:下面會從淺到深,淡淡在閱讀源碼過程中自己的理解。分拆子頁面后,每一個子頁面對應一個文件。總結上面就是最早版本的源碼,很簡潔的使用了等其目的也很簡單簡化相關生態的繁瑣邏輯參考源碼地址 ??dva的思想還是很不錯的,大大提升了開發效率,dva集成了Redux以及Redux的中間件Redux-saga,以及React-router等等。得益于Redux的狀態管理,以及Redux-saga中...
摘要:下面會從淺到深,淡淡在閱讀源碼過程中自己的理解。分拆子頁面后,每一個子頁面對應一個文件。總結上面就是最早版本的源碼,很簡潔的使用了等其目的也很簡單簡化相關生態的繁瑣邏輯參考源碼地址 ??dva的思想還是很不錯的,大大提升了開發效率,dva集成了Redux以及Redux的中間件Redux-saga,以及React-router等等。得益于Redux的狀態管理,以及Redux-saga中...
摘要:下面會從淺到深,淡淡在閱讀源碼過程中自己的理解。分拆子頁面后,每一個子頁面對應一個文件。總結上面就是最早版本的源碼,很簡潔的使用了等其目的也很簡單簡化相關生態的繁瑣邏輯參考源碼地址 ??dva的思想還是很不錯的,大大提升了開發效率,dva集成了Redux以及Redux的中間件Redux-saga,以及React-router等等。得益于Redux的狀態管理,以及Redux-saga中...
閱讀 1567·2021-11-24 09:39
閱讀 1053·2021-11-22 15:11
閱讀 2174·2021-11-19 11:35
閱讀 1630·2021-09-13 10:37
閱讀 2459·2021-09-03 10:47
閱讀 2145·2021-08-30 09:47
閱讀 1635·2021-08-20 09:39
閱讀 2907·2019-08-30 14:13