1. ホーム
  2. php

[解決済み] foreach、array_mapとlambda、array_mapとstatic関数の性能比較

2022-05-10 05:27:45

質問

配列を別の配列に変換するために使用されるこれら3つのアプローチには、どのようなパフォーマンスの違いがあるのでしょうか(あるとすれば)?

  1. 使用方法 foreach
  2. 使用中 array_map をラムダ/クロージャ関数で使用する
  3. 使用方法 array_map を「静的」な関数/メソッドで使用する
  4. 他の方法はないのでしょうか?

私自身を明確にするために、すべての例を見てみましょう。

$numbers = range(0, 1000);


フォアグラウンド

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


ラムダを使ったマップ

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


static'関数付きマップ、文字列参照として渡される

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


他に方法はないのでしょうか?実際にお聞かせください。 すべて 上記のケースの違いや、なぜ他のケースの代わりに1つを使用する必要があるのかのインプットを教えてください。

どのように解決するのですか?

参考までに、投稿者がやっていなかったので、ベンチマークだけやってみました。PHP 5.3.10 + XDebugで動作しています。

UPDATE 2015-01-22 XDebugなし、より新しいPHPバージョンでの追加結果については、以下のmcfedrの回答と比較してください。


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";
}


1Mの数字で十数回の試行でかなり安定した結果を得ることができます。

  • 実行時間 0.7 秒
  • クロージャのマップ:3.4秒
  • 関数名でマップ:1.2秒

map on closureの速度が出ないのは、closureが毎回評価されている可能性があるためと仮定して、こんなテストもしてみました。


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

  return array_map($closure, $numbers);
}

しかし、結果は同じで、クロージャが一度だけ評価されることが確認されました。

2014-02-02 UPDATE: オペコードダンプ

3つのコールバックのオペコードダンプを紹介します。まず 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