携帯電話関連の Web サイトを運営しているとします。ユーザーは、いくつかのパラメーター (オペレーティング システム、画面解像度、カメラのピクセルなど) を指定して、必要な携帯電話をフィルタリングできるとします。ただし、携帯電話には多くのパラメータがあり、携帯電話ごとにパラメータが大きく異なるため、パラメータ テーブルの構造は通常、水平テーブル (1 パラメータが 1 列) ではなく、垂直テーブル (1 パラメータが 1 行) になります。 、結果を取得するために複数のパラメータが使用されます。通常は、それぞれのパラメータを取得して結果を取得し、共通部分を取得します。
各パラメータには約 1,000 個の製品 ID (int) が含まれると想定し、これを前提としてデータをシミュレートして生成します。
<?php $rand = function() { $result = array(); for ($i = 0; $i < 1000; $i++) { $result[] = mt_rand(1, 10000); } return $result; }; $param_a = $rand(); $param_b = $rand(); ?>
注: テスト データ セットが小さすぎる場合、結論に一貫性がなくなる可能性があります。
まず、PHP の組み込みメソッド array_intersect によって達成されるパフォーマンスを見てみましょう:
<?php $time = microtime(true); $result = array_intersect($param_a, $param_b); $time = microtime(true) - $time; echo "array_intersect: {$time}\n"; ?>
最適化する前に、array_intersect の特別な機能をいくつか見てみましょう:
<?php $param_a = array(1, 2, 2); $param_b = array(1, 2, 3); var_dump( array_intersect($param_a, $param_b), array_intersect($param_b, $param_a) ); ?>
つまり、最初の配列パラメーターに重複した要素がある場合、array_intersect は条件を満たすすべての重複した要素を返します。 array_intersect を書き換えるときは、これらの関数と互換性があることが最善です。
カスタム メソッド int_array_intersect によって達成されるパフォーマンスを見てみましょう:
<?php function int_array_intersect() { if (func_num_args() < 2) { trigger_error('param error', E_USER_ERROR); } $args = func_get_args(); foreach ($args AS $arg) { if (!is_array($arg)) { trigger_error('param error', E_USER_ERROR); } } $intersect = function($a, $b) { $result = array(); $length_a = count($a); $length_b = count($b); for ($i = 0, $j = 0; $i < $length_a && $j < $length_b; null) { if($a[$i] < $b[$j] && ++$i) { continue; } if($a[$i] > $b[$j] && ++$j) { continue; } $result[] = $a[$i]; if (isset($a[$next = $i + 1]) && $a[$next] != $a[$i]) { ++$j; } ++$i; } return $result; }; $result = array_shift($args); sort($result); foreach ($args as $arg) { sort($arg); $result = $intersect($result, $arg); } return $result; } $time = microtime(true); $result = int_array_intersect($param_a, $param_b); $time = microtime(true) - $time; echo "int_array_intersect: {$time}\n"; ?>
直感的には、組み込み関数はカスタム関数よりも高速であると考えられますが、この場合はその逆です。
なぜですか?その理由は、int_array_intersect は整数を操作するのに対し、array_intersect は文字列を操作するため、整数を渡すと、時間のかかる「(string)」型変換操作が実行されるためです。
注: テスト結果は PHP5.3.5 に基づいており、バージョンが異なると結論が異なる場合があります