首頁 後端開發 php教程 PHP源代码数组统计count分析_php技巧

PHP源代码数组统计count分析_php技巧

May 17, 2016 am 09:16 AM
count 數組統計

zend给php的所有变量都用结构的方式去保存,而字符串的保存和数组的保存也是不同的,数组采用的是hash表的方式去保存(大家知道hash保存的地址有效的减少冲突-hash散列表的概念你懂的),而在php中的结构体上表现如下:

复制代码 代码如下:

//文件1:zend/zend.h
/*
* zval
*/
typedef struct _zval_struct zval;
...
typedef union _zvalue_value {
long lval; /* long value */
double dval; /* double value */
struct {
char *val;
int len;
} str;
HashTable *ht; /* hash table value */
zend_object_value obj;
} zvalue_value;

struct _zval_struct {
/* Variable information */
zvalue_value value; /* value */
zend_uint refcount__gc;
zend_uchar type; /* active type */
zend_uchar is_ref__gc;
};
//hash表的结构如下
//文件2:zend/zend_hash.h
typedef struct _hashtable {
uint nTableSize;
uint nTableMask;
uint nNumOfElements;
ulong nNextFreeElement;
Bucket *pInternalPointer; /* Used for element traversal */
Bucket *pListHead;
Bucket *pListTail;
Bucket **arBuckets;
dtor_func_t pDestructor;
zend_bool persistent;
unsigned char nApplyCount;
zend_bool bApplyProtection;
#if ZEND_DEBUG
int inconsistent;
#endif
}
HashTable;

一般的变量(字符串)在使用strlen获取长度的时候,其实获取的就是zvalue_value.str这个结构中的len属性,效率上O(1)次,特别说明的一点是:strlen在php中并没有核心的实现,而是在使用了zend中的宏定义来获取:

复制代码 代码如下:

//文件3:zend/zend_operators.php
#define Z_STRLEN(zval) (zval).value.str.len
...
#define Z_STRLEN_P(zval_p) Z_STRLEN(*zval_p)
...
#define Z_STRLEN_PP(zval_pp) Z_STRLEN(**zval_pp)

而对于数组的count操作,其实有两种结果,在count 的api中也提到了第二个参数mode《http://www.php.net/manual/en/function.count.php》,这个mode参数指明了,是否需要重新统计,而它的重新统计将会遍历一次数组,效率上是O(N)[N:长度],默认情况下是不重新统计,那这个时候将会直接输出hashtable中的nNumOfElements,此时的效率也是O(1)次:count代码如下:
复制代码 代码如下:

//文件4:ext/standard/array.c
PHP_FUNCTION(count)
{
zval *array;
long mode = COUNT_NORMAL;

if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z|l", &array, &mode) == FAILURE) {
return;
}

switch (Z_TYPE_P(array)) {
case IS_NULL:
RETURN_LONG(0);
break;
case IS_ARRAY:
RETURN_LONG (php_count_recursive (array, mode TSRMLS_CC));
break;
.....

//php_count_recursive的实现
static int php_count_recursive(zval *array, long mode TSRMLS_DC) /* {{{ */
{
long cnt = 0;
zval **element;

if (Z_TYPE_P(array) == IS_ARRAY) {
//错误处理
if (Z_ARRVAL_P(array)->nApplyCount > 1) {
php_error_docref(NULL TSRMLS_CC, E_WARNING, "recursion detected");
return 0;
}
//通过zend_hash_num_elements直接获得长度
cnt = zend_hash_num_elements(Z_ARRVAL_P(array));

//如果指定了需要重新统计,则会进入一次循环统计
if (mode == COUNT_RECURSIVE) {
HashPosition pos;

for (zend_hash_internal_pointer_reset_ex(Z_ARRVAL_P(array), &pos);
zend_hash_get_current_data_ex(Z_ARRVAL_P(array), (void **) &element, &pos) == SUCCESS;
zend_hash_move_forward_ex(Z_ARRVAL_P(array), &pos)
) {
Z_ARRVAL_P(array)->nApplyCount++;
cnt += php_count_recursive(*element, COUNT_RECURSIVE TSRMLS_CC);
Z_ARRVAL_P(array)->nApplyCount--;
}
}
}

return cnt;
}

//文件5:zend/zend_hash.c
//zend_hash_num_elements的实现
ZEND_API int zend_hash_num_elements(const HashTable *ht)
{
IS_CONSISTENT(ht);

return ht->nNumOfElements;
}
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

counta和count的區別 counta和count的區別 Nov 20, 2023 am 10:01 AM

Count函數用於計算指定範圍內數字的個數。它忽略文字、邏輯值和空值,但會將空白儲存格計算在內,Count函數只計算包含實際數字的儲存格數量。而CountA函數用於計算指定範圍內非空單元格的個數。它不僅計算包含實際數字的儲存格,還計算包含文字、邏輯值和公式等非空白儲存格的數量。

如何使用PHP中的array_count_values函數統計數組中每個值出現的次數 如何使用PHP中的array_count_values函數統計數組中每個值出現的次數 Jun 26, 2023 pm 02:12 PM

在PHP中,我們經常需要對陣列進行各種操作,統計數組中每個值出現的次數是其中一個常見的需求。 PHP提供了一個簡單且有效率的函數-array_count_values,可以輕鬆完成這個任務。 array_count_values函數的語法如下:array_count_values(array$array)這個函數接收一個陣列參數,傳回一個新的數組,其中包含原

MySQL關於Count函數的用法差異總結 MySQL關於Count函數的用法差異總結 Nov 29, 2022 pm 04:47 PM

本文為大家介紹MySQL關於Count函數的用法區別,統計數據的需求在我們日常開發中是非常容易遇到了,MySQL也支援多種的計算的函數,接下來我們來看看他們之間有什麼區別,以及他們是否存在一些坑。

PHP函數介紹—array_count_values(): 統計數組中每個元素的出現次數 PHP函數介紹—array_count_values(): 統計數組中每個元素的出現次數 Jul 25, 2023 pm 07:18 PM

PHP函數介紹—array_count_values():統計數組中每個元素的出現次數在PHP的開發中,經常會遇到需要對數組中的元素進行統計的情況。 PHP提供了一些方便的函數來幫助我們實現這個目標,其中之一就是array_count_values()函數。 array_count_values()函數能夠統計數組中每個元素的出現次數,並傳回關聯數組,其中

php中count函數的用法是什麼 php中count函數的用法是什麼 Sep 09, 2021 pm 05:08 PM

php中count函數的用法是:【count(array,mode);】,其中參數array規定要計數的數組,參數mode規定函數的模式。

mysql踩坑之count distinct多列問題怎麼解決 mysql踩坑之count distinct多列問題怎麼解決 Jun 03, 2023 am 10:49 AM

重複的測試資料庫如下所示:CREATETABLE`test_distinct`(`id`int(11)NOTNULLAUTO_INCREMENT,`a`varchar(50)CHARACTERSETutf8DEFAULTNULL,`b`varchar(50)CHARACTERSETutf8DEFAULTN InnoDBAUTO_INCREMENT=1DEFAULTCHARSET=latin1;表內測試資料如下,現在我們需要統計這三列去重後的列

學習使用PHP count()函數計算陣列元素個數 學習使用PHP count()函數計算陣列元素個數 Jun 27, 2023 am 10:37 AM

PHP是一種非常流行的伺服器端程式語言,它具有強大的開發能力,尤其是在陣列操作方面。 PHP中的陣列(array)被廣泛使用,具有儲存和管理多個資料項的能力。在對數組進行操作時,經常需要計算數組中元素的個數,此時可以透過使用PHP中的count()函數來實現。 count()函數是PHP內建的計算陣列元素個數的函數。在使用函數時,只需要將數組作為函數的參數傳

java Count怎麼計算流中的元素 java Count怎麼計算流中的元素 May 11, 2023 pm 04:07 PM

說明1、count是終端機操作,可以統計stream流中的元素總數,傳回值為long類型。 2、count()傳回流中元素的計數。這是歸納的特殊情況(歸納運算採用一系列輸入元素,透過重複應用組合運算將其組合成一個總結結果)。這是終端操作,可能會產生結果和副作用。執行終端操作後,管道被視為消耗,無法再利用。實例//驗證list中string是否有以a開頭的,匹配到第一個,即傳回truebooleananyStartsWithA=stringCollection.stream().anyMatch((s

See all articles