亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > PHP > 正文

PHP中array_keys和array_unique函數源碼的分析

2024-05-04 23:43:18
字體:
來源:轉載
供稿:網友
本文從array_keys和array_unique的源碼分析出函數的性能,并給出了優化建議,十分不錯的文章,有需要的小伙伴可以參考下
 

性能分析

從運行性能上分析,看看下面的測試代碼:

$test=array();for($run=0; $run<10000; $run++)$test[]=rand(0,100);$time=microtime(true);$out = array_unique($test);$time=microtime(true)-$time;echo 'Array Unique: '.$time."/n";$time=microtime(true);$out=array_keys(array_flip($test));$time=microtime(true)-$time;echo 'Keys Flip: '.$time."/n";$time=microtime(true);$out=array_flip(array_flip($test));$time=microtime(true)-$time;echo 'Flip Flip: '.$time."/n";

運行結果如下:

 

從上圖可以看到,使用array_unique函數需要0.069s;使用array_flip后再使用array_keys函數需要0.00152s;使用兩次array_flip函數需要0.00146s。

測試結果表明,使用array_flip后再調用array_keys函數比array_unique函數快。那么,具體原因是什么呢?讓我們看看在PHP底層,這兩個函數是怎么實現的。

源碼分析

/* {{{ proto array array_keys(array input [, mixed search_value[, bool strict]])  Return just the keys from the input array, optionally only for the specified       search_value */PHP_FUNCTION(array_keys){  //變量定義  zval *input,        /* Input array */     *search_value = NULL,  /* Value to search for */     **entry,        /* An entry in the input array */      res,          /* Result of comparison */     *new_val;        /* New value */  int  add_key;        /* Flag to indicate whether a key should be added */  char *string_key;      /* String key */  uint  string_key_len;  ulong num_key;        /* Numeric key */  zend_bool strict = 0;    /* do strict comparison */  HashPosition pos;  int (*is_equal_func)(zval *, zval *, zval * TSRMLS_DC) = is_equal_function;  //程序解析參數  if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "a|zb", &input, &search_value, &strict) == FAILURE) {    return;  }  // 如果strict是true,則設置is_equal_func為is_identical_function,即全等比較  if (strict) {    is_equal_func = is_identical_function;  }  /* 根據search_vale初始化返回的數組大小 */  if (search_value != NULL) {    array_init(return_value);  } else {    array_init_size(return_value, zend_hash_num_elements(Z_ARRVAL_P(input)));  }  add_key = 1;  /* 遍歷輸入的數組參數,然后添加鍵值到返回的數組 */  zend_hash_internal_pointer_reset_ex(Z_ARRVAL_P(input), &pos);//重置指針  //循環遍歷數組  while (zend_hash_get_current_data_ex(Z_ARRVAL_P(input), (void **)&entry, &pos) == SUCCESS) {    // 如果search_value不為空    if (search_value != NULL) {      // 判斷search_value與當前的值是否相同,并將比較結果保存到add_key變量      is_equal_func(&res, search_value, *entry TSRMLS_CC);      add_key = zval_is_true(&res);    }    if (add_key) {      // 創建一個zval結構體      MAKE_STD_ZVAL(new_val);      // 根據鍵值是字符串還是整型數字將值插入到return_value中      switch (zend_hash_get_current_key_ex(Z_ARRVAL_P(input), &string_key, &string_key_len, &num_key, 1, &pos)) {        case HASH_KEY_IS_STRING:          ZVAL_STRINGL(new_val, string_key, string_key_len - 1, 0);          // 此函數負責將值插入到return_value中,如果鍵值已存在,則使用新值更新對應的值,否則直接插入          zend_hash_next_index_insert(Z_ARRVAL_P(return_value), &new_val, sizeof(zval *), NULL);          break;        case HASH_KEY_IS_LONG:          Z_TYPE_P(new_val) = IS_LONG;          Z_LVAL_P(new_val) = num_key;          zend_hash_next_index_insert(Z_ARRVAL_P(return_value), &new_val, sizeof(zval *), NULL);          break;      }    }    // 移動到下一個    zend_hash_move_forward_ex(Z_ARRVAL_P(input), &pos);  }}/* }}} */

以上是array_keys函數底層的源碼。為方便理解,筆者添加了一些中文注釋。如果需要查看原始代碼,可以點擊查看。這個函數的功能就是新建一個臨時數組,然后將鍵值對重新復制到新的數組,如果復制過程中有重復的鍵值出現,那么就用新的值替換。這個函數的主要步驟是地57和63行調用的zend_hash_next_index_insert函數。該函數將元素插入到數組中,如果出現重復的值,則使用新的值更新原鍵值指向的值,否則直接插入,時間復雜度是O(n)。

/* {{{ proto array array_flip(array input)  Return array with key <-> value flipped */PHP_FUNCTION(array_flip){  // 定義變量  zval *array, **entry, *data;  char *string_key;  uint str_key_len;  ulong num_key;  HashPosition pos;  // 解析數組參數  if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "a", &array) == FAILURE) {    return;  }  // 初始化返回數組  array_init_size(return_value, zend_hash_num_elements(Z_ARRVAL_P(array)));  // 重置指針  zend_hash_internal_pointer_reset_ex(Z_ARRVAL_P(array), &pos);  // 遍歷每個元素,并執行鍵<->值交換操作  while (zend_hash_get_current_data_ex(Z_ARRVAL_P(array), (void **)&entry, &pos) == SUCCESS) {    // 初始化一個結構體    MAKE_STD_ZVAL(data);    // 將原數組的值賦值為新數組的鍵    switch (zend_hash_get_current_key_ex(Z_ARRVAL_P(array), &string_key, &str_key_len, &num_key, 1, &pos)) {      case HASH_KEY_IS_STRING:        ZVAL_STRINGL(data, string_key, str_key_len - 1, 0);        break;      case HASH_KEY_IS_LONG:        Z_TYPE_P(data) = IS_LONG;        Z_LVAL_P(data) = num_key;        break;    }    // 將原數組的鍵賦值為新數組的值,如果有重復的,則使用新值覆蓋舊值    if (Z_TYPE_PP(entry) == IS_LONG) {      zend_hash_index_update(Z_ARRVAL_P(return_value), Z_LVAL_PP(entry), &data, sizeof(data), NULL);    } else if (Z_TYPE_PP(entry) == IS_STRING) {      zend_symtable_update(Z_ARRVAL_P(return_value), Z_STRVAL_PP(entry), Z_STRLEN_PP(entry) + 1, &data, sizeof(data), NULL);    } else {      zval_ptr_dtor(&data); /* will free also zval structure */      php_error_docref(NULL TSRMLS_CC, E_WARNING, "Can only flip STRING and INTEGER values!");    }    // 下一個    zend_hash_move_forward_ex(Z_ARRVAL_P(array), &pos);  }}/* }}} */

上面就是是array_flip函數的源碼。點擊鏈接查看原始代碼。這個函數主要的做的事情就是創建一個新的數組,遍歷原數組。在26行開始將原數組的值賦值為新數組的鍵,然后在37行開始將原數組的鍵賦值為新數組的值,如果有重復的,則使用新值覆蓋舊值。整個函數的時間復雜度也是O(n)。因此,使用了array_flip之后再使用array_keys的時間復雜度是O(n)。

接下來,我們看看array_unique函數的源碼。點擊鏈接查看原始代碼。

/* {{{ proto array array_unique(array input [, int sort_flags])  Removes duplicate values from array */PHP_FUNCTION(array_unique){  // 定義變量  zval *array, *tmp;  Bucket *p;  struct bucketindex {    Bucket *b;    unsigned int i;  };  struct bucketindex *arTmp, *cmpdata, *lastkept;  unsigned int i;  long sort_type = PHP_SORT_STRING;  // 解析參數  if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "a|l", &array, &sort_type) == FAILURE) {    return;  }  // 設置比較函數  php_set_compare_func(sort_type TSRMLS_CC);  // 初始化返回數組  array_init_size(return_value, zend_hash_num_elements(Z_ARRVAL_P(array)));  // 將值拷貝到新數組  zend_hash_copy(Z_ARRVAL_P(return_value), Z_ARRVAL_P(array), (copy_ctor_func_t) zval_add_ref, (void *)&tmp, sizeof(zval*));  if (Z_ARRVAL_P(array)->nNumOfElements <= 1) {  /* 什么都不做 */    return;  }  /* 根據target_hash buckets的指針創建數組并排序 */  arTmp = (struct bucketindex *) pemalloc((Z_ARRVAL_P(array)->nNumOfElements + 1) * sizeof(struct bucketindex), Z_ARRVAL_P(array)->persistent);  if (!arTmp) {    zval_dtor(return_value);    RETURN_FALSE;  }  for (i = 0, p = Z_ARRVAL_P(array)->pListHead; p; i++, p = p->pListNext) {    arTmp[i].b = p;    arTmp[i].i = i;  }  arTmp[i].b = NULL;  // 排序  zend_qsort((void *) arTmp, i, sizeof(struct bucketindex), php_array_data_compare TSRMLS_CC);  /* 遍歷排序好的數組,然后刪除重復的元素 */  lastkept = arTmp;  for (cmpdata = arTmp + 1; cmpdata->b; cmpdata++) {    if (php_array_data_compare(lastkept, cmpdata TSRMLS_CC)) {      lastkept = cmpdata;    } else {      if (lastkept->i > cmpdata->i) {        p = lastkept->b;        lastkept = cmpdata;      } else {        p = cmpdata->b;      }      if (p->nKeyLength == 0) {        zend_hash_index_del(Z_ARRVAL_P(return_value), p->h);      } else {        if (Z_ARRVAL_P(return_value) == &EG(symbol_table)) {          zend_delete_global_variable(p->arKey, p->nKeyLength - 1 TSRMLS_CC);        } else {          zend_hash_quick_del(Z_ARRVAL_P(return_value), p->arKey, p->nKeyLength, p->h);        }      }    }  }  pefree(arTmp, Z_ARRVAL_P(array)->persistent);}/* }}} */

可以看到,這個函數初始化一個新的數組,然后將值拷貝到新數組,然后在45行調用排序函數對數組進行排序,排序的算法是zend引擎的塊樹排序算法。接著遍歷排序好的數組,刪除重復的元素。整個函數開銷最大的地方就在調用排序函數上,而快排的時間復雜度是O(nlogn),因此,該函數的時間復雜度是O(nlogn)。

結論

因為array_unique底層調用了快排算法,加大了函數運行的時間開銷,導致整個函數的運行較慢。這就是為什么array_keys比array_unique函數更快的原因。



注:相關教程知識閱讀請移步到PHP教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲激情电影中文字幕| 热99在线视频| 欧美视频免费在线| 久久男人av资源网站| 日韩视频欧美视频| 日韩美女福利视频| 久久精品99久久久香蕉| 国产一区二区三区视频在线观看| 日韩欧美视频一区二区三区| 96sao精品视频在线观看| 91精品视频网站| 久久久久国产精品免费| 国产欧洲精品视频| 性欧美视频videos6一9| 欧美成人午夜激情| 亚洲最新av在线网站| 国产精品久久久| 91在线直播亚洲| 亚洲免费一在线| 日韩欧美高清在线视频| 国产日韩中文字幕| 欧美激情视频一区二区| 国产精品偷伦一区二区| 国产精品69精品一区二区三区| 第一福利永久视频精品| 欧美高清视频一区二区| 国产成人在线一区| 欧美午夜激情在线| 国产午夜精品一区理论片飘花| 日韩国产欧美精品一区二区三区| 国产亚洲aⅴaaaaaa毛片| 久久久视频精品| 日韩av在线免费观看一区| 久久99精品国产99久久6尤物| 日本aⅴ大伊香蕉精品视频| 国产欧美在线播放| 毛片精品免费在线观看| 成人淫片在线看| 欧美区二区三区| 国产精品伦子伦免费视频| 91经典在线视频| 久热精品视频在线| 国产91在线播放九色快色| 欧美日韩国产一区在线| 日韩av在线直播| 中文字幕日韩欧美在线视频| 欧美性在线视频| 欧美亚洲视频在线看网址| 亚洲欧美日韩另类| 国产深夜精品福利| 欧美视频在线免费| 国产va免费精品高清在线| 久久精品夜夜夜夜夜久久| 国产精品一区=区| 国内精品久久久久久| 亚洲国产精品久久久久秋霞不卡| 欧美在线视频免费| 欧美亚洲午夜视频在线观看| 高清日韩电视剧大全免费播放在线观看| 亚洲一区二区少妇| 粉嫩老牛aⅴ一区二区三区| 欧美国产激情18| 日韩精品高清视频| 国内精品久久久久久久| 欧美精品videos| 国产91ⅴ在线精品免费观看| 久久影院免费观看| 日韩欧美综合在线视频| 亚洲美女视频网站| 国产一区红桃视频| 92看片淫黄大片欧美看国产片| 亚洲系列中文字幕| 欧美成人午夜免费视在线看片| 国语自产偷拍精品视频偷| 成人综合网网址| 久久亚洲国产精品| 国产精品福利小视频| 久久久av一区| 555www成人网| 亚洲韩国日本中文字幕| 亚洲精品456在线播放狼人| 日韩免费观看网站| 日韩av快播网址| 日韩激情av在线免费观看| 亚洲人成网站在线播| 国产成人精品免高潮在线观看| 国产美女主播一区| 国产成人精品免费久久久久| 国产在线视频2019最新视频| 欧洲精品久久久| 成人久久18免费网站图片| 日韩精品视频免费专区在线播放| 日韩激情av在线免费观看| 午夜精品www| 最近2019年中文视频免费在线观看| 欧美日韩国产精品一区| 日韩av在线影院| 国产精品免费在线免费| 国产99久久精品一区二区| 97在线观看视频国产| 亚洲专区国产精品| 国产精品成人一区二区| 国产成人一区三区| 另类色图亚洲色图| 日韩久久精品电影| 亚洲视频国产视频| 欧美日韩精品在线| 中文字幕日韩高清| 欧美高清视频在线播放| 欧美国产中文字幕| 国模视频一区二区三区| 深夜福利日韩在线看| 精品国产91久久久久久老师| 国产精品久久久久一区二区| 欧美怡春院一区二区三区| 少妇激情综合网| 精品视频—区二区三区免费| 国内精品视频一区| 欧美另类69精品久久久久9999| 国产欧美精品日韩| 日韩欧美在线网址| 亚洲成人亚洲激情| 欧美性猛交xxxx黑人猛交| 亚洲色图13p| 91tv亚洲精品香蕉国产一区7ujn| 亚洲欧洲在线看| 77777亚洲午夜久久多人| 色偷偷av亚洲男人的天堂| 欧美精品久久久久久久| 国产aⅴ夜夜欢一区二区三区| 正在播放欧美一区| 最近2019中文字幕大全第二页| 亚洲成avwww人| 亚洲乱亚洲乱妇无码| 国产z一区二区三区| 欧美福利视频网站| 亚洲sss综合天堂久久| 91网站免费观看| 2019精品视频| 成人免费自拍视频| 久久精品福利视频| 国产日韩在线亚洲字幕中文| 狠狠躁18三区二区一区| 成人免费黄色网| 欧美激情国产高清| 在线播放国产精品| 久久综合国产精品台湾中文娱乐网| 国产剧情日韩欧美| 国产大片精品免费永久看nba| 亚洲男人天堂九九视频| 国产精品视频区1| 国产精品久久久久久久av电影| 欧美尤物巨大精品爽| 亚洲a一级视频| 日韩国产激情在线| 精品国内亚洲在观看18黄| 成人中心免费视频| 欧美国产日韩在线| 国产精品视频久久久久| 最近免费中文字幕视频2019| 黄色一区二区三区| 亚洲电影免费观看| 欧美日韩一区二区免费视频| 亚洲高清在线观看|