这三种方法都用于将一个数组转换为另一个数组,它们之间的性能差异(如果有)是什么?

  1. 使用foreach
  2. array_map与lambda/闭包函数一起使用
  3. 使用带有"静态"功能/方法的array_map
  4. 还有没有别的办法?

为了清楚起见,让我们看一下示例,所有示例都在执行相同的操作-将数字数组乘以10:

$numbers = range(0, 1000);

Foreach

$result = array();
foreach ($numbers as $number) {
    $result[] = $number * 10;
}
return $result;

Map with lambda

return array_map(function($number) {
    return $number * 10;
}, $numbers);

Map with 'static' function, passed as string reference

function tenTimes($number) {
    return $number * 10;
}
return array_map('tenTimes', $numbers);

还有其他方法吗?我很高兴听到上面的all个 case 之间的差异,以及为什么应该使用一个而不是其他 case 的任何输入.

推荐答案

顺便说一下,我刚做了基准测试,因为Poster没有做.在PHP 5.3.10+XDebug上运行.

更新2015-01-22请与下面mcfedr的答案进行比较,以了解没有XDebug和更新的PHP版本的其他结果.


function lap($func) {
  $t0 = microtime(1);
  $numbers = range(0, 1000000);
  $ret = $func($numbers);
  $t1 = microtime(1);
  return array($t1 - $t0, $ret);
}

function useForeach($numbers)  {
  $result = array();
  foreach ($numbers as $number) {
      $result[] = $number * 10;
  }
  return $result;
}

function useMapClosure($numbers) {
  return array_map(function($number) {
      return $number * 10;
  }, $numbers);
}

function _tenTimes($number) {
    return $number * 10;
}

function useMapNamed($numbers) {
  return array_map('_tenTimes', $numbers);
}

foreach (array('Foreach', 'MapClosure', 'MapNamed') as $callback) {
  list($delay,) = lap("use$callback");
  echo "$callback: $delay\n";
}

我在十几次try 中获得了100万个数字的一致结果:

  • Foreach:0.7秒
  • 关闭时的贴图:3.4秒
  • 映射到函数名称:1.2秒.

假设闭包时 map 的平淡速度是由每次可能被判断的闭包引起的,我也这样测试:


function useMapClosure($numbers) {
  $closure = function($number) {
    return $number * 10;
  };

  return array_map($closure, $numbers);
}

但结果是一致的,证实了闭包只判断了一次.

2014-02-02更新:操作码转储

以下是三个回调的操作码转储.前useForeach()名:



compiled vars:  !0 = $numbers, !1 = $result, !2 = $number
line     # *  op                           fetch          ext  return  operands
---------------------------------------------------------------------------------
  10     0  >   EXT_NOP                                                  
         1      RECV                                                     1
  11     2      EXT_STMT                                                 
         3      INIT_ARRAY                                       ~0      
         4      ASSIGN                                                   !1, ~0
  12     5      EXT_STMT                                                 
         6    > FE_RESET                                         $2      !0, ->15
         7  > > FE_FETCH                                         $3      $2, ->15
         8  >   OP_DATA                                                  
         9      ASSIGN                                                   !2, $3
  13    10      EXT_STMT                                                 
        11      MUL                                              ~6      !2, 10
        12      ASSIGN_DIM                                               !1
        13      OP_DATA                                                  ~6, $7
  14    14    > JMP                                                      ->7
        15  >   SWITCH_FREE                                              $2
  15    16      EXT_STMT                                                 
        17    > RETURN                                                   !1
  16    18*     EXT_STMT                                                 
        19*   > RETURN                                                   null

然后是useMapClosure()


compiled vars:  !0 = $numbers
line     # *  op                           fetch          ext  return  operands
---------------------------------------------------------------------------------
  18     0  >   EXT_NOP                                                  
         1      RECV                                                     1
  19     2      EXT_STMT                                                 
         3      EXT_FCALL_BEGIN                                          
         4      DECLARE_LAMBDA_FUNCTION                                  '%00%7Bclosure%7D%2Ftmp%2Flap.php0x7f7fc1424173'
  21     5      SEND_VAL                                                 ~0
         6      SEND_VAR                                                 !0
         7      DO_FCALL                                      2  $1      'array_map'
         8      EXT_FCALL_END                                            
         9    > RETURN                                                   $1
  22    10*     EXT_STMT                                                 
        11*   > RETURN                                                   null

以及它所称的终结:


compiled vars:  !0 = $number
line     # *  op                           fetch          ext  return  operands
---------------------------------------------------------------------------------
  19     0  >   EXT_NOP                                                  
         1      RECV                                                     1
  20     2      EXT_STMT                                                 
         3      MUL                                              ~0      !0, 10
         4    > RETURN                                                   ~0
  21     5*     EXT_STMT                                                 
         6*   > RETURN                                                   null

useMapNamed()函数:


compiled vars:  !0 = $numbers
line     # *  op                           fetch          ext  return  operands
---------------------------------------------------------------------------------
  28     0  >   EXT_NOP                                                  
         1      RECV                                                     1
  29     2      EXT_STMT                                                 
         3      EXT_FCALL_BEGIN                                          
         4      SEND_VAL                                                 '_tenTimes'
         5      SEND_VAR                                                 !0
         6      DO_FCALL                                      2  $0      'array_map'
         7      EXT_FCALL_END                                            
         8    > RETURN                                                   $0
  30     9*     EXT_STMT                                                 
        10*   > RETURN                                                   null

它调用的命名函数是_tenTimes():


compiled vars:  !0 = $number
line     # *  op                           fetch          ext  return  operands
---------------------------------------------------------------------------------
  24     0  >   EXT_NOP                                                  
         1      RECV                                                     1
  25     2      EXT_STMT                                                 
         3      MUL                                              ~0      !0, 10
         4    > RETURN                                                   ~0
  26     5*     EXT_STMT                                                 
         6*   > RETURN                                                   null

Php相关问答推荐

显示根据WooCommerce单变量产品中的数量计算得出的汇总

Laravel 11发送邮箱时奇怪的未定义数组键名称

未在moodle上运行的计划任务

如何优化-PHP 7.3.14+Laravel 6

Mysqli_stmt::Execute():结果字段元数据中存在过早的EOF

在WooCommerce管理订单列表中显示项目总权重

在特定订单状态更改时自定义WooCommerce订单发货项目

使用php ZipArhive类将Zip压缩文件分成多个部分

限制某些产品只能在WooCommerce的特定邮政编码/邮政编码范围内发货

函数存储到变量中

根据WooCommerce中的自定义用户元数据值更改用户角色

WordPress:今天的计数';修改和发布的帖子

$this->;db->;update();CodIgnitor中的功能不工作

单个产品页面 WooCommerce 订阅价格字符串的更改

PHP API - 本地主机中的 SSL 证书问题

使用 Laravel 和 Vue 进行图像表单验证

$_SERVER 超全局变量在 PHP 中是否保证可写?

优化后的标题:如何在WooCommerce我的帐户编辑表单中添加账单字段

为什么在 phpunit 的开头测试 PHP_VERSION?

为什么 8.0 之前的 PHP 版本拒绝命名空间Random\Namespace作为无效语法?