最近被问到一个函数count
1、count("123456") 对应的输出是什么?
2、count(null) 对应的输出是什么?
以前没有认真的考虑,只是心里有一个印象那就是count()函数是计算php数组的个数的。后来下来仔细看了看手册,原来发现:
1、count函数是统计数组里所有元素的个数或者对象的属性的个数的,函数原型如下:
int count(mixed $var[, $mode=COUNT_NORMAL);
对于对象,如果安装了SPL,可以通过Countable接口调用count,该接口只有一个Countable::count(),此函数返回count的返回值。
2、当设置$mode=1时,会递归对数组进行计算,但是识别不了无限递归。
3、当$var不是array或者object(实现了Countalbe::count函数)时:
count函数将返回1,但当$mixed=null 时,将返回0
count函数实现的源码是:
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;
case IS_OBJECT: {
#ifdef HAVE_SPL
zval *retval;
#endif
/* first, we check if the handler is defined */
if (Z_OBJ_HT_P(array)->count_elements) {
RETVAL_LONG();
if (SUCCESS == Z_OBJ_HT(*array)->count_elements(array, &Z_LVAL_P(return_value) TSRMLS_CC)) {
return;
}
}
#ifdef HAVE_SPL //如果安装了 SPL
/* if not and the object implements Countable we call its count() method */
if (Z_OBJ_HT_P(array)->get_class_entry && instanceof_function(Z_OBJCE_P(array), spl_ce_Countable TSRMLS_CC)) {
zend_call_method_with_0_params(&array, NULL, NULL, "count", &retval);
if (retval) {
convert_to_long_ex(&retval);
RETVAL_LONG(Z_LVAL_P(retval));
zval_ptr_dtor(&retval);
}
return;
}
#endif
}
default://其他类型返回1
RETURN_LONG();
break;
}
}
以上可以看出,调用count函数时,当$mixed为数组时会调用php_count_recursive 函数
(php_count_recursive,当不是递归获取元素个数时,返回数组的nNumOfElements,时间复杂度为N(1),只有mode=1时才会递归计算元素个数);
当$mixed是null时,返回0
默认都是返回1
结论:
想了解一些PHP函数的时候还是多看看PHP手册,以前主要是通过W3C上面了解的。