FWIW ฉันเพิ่งทำมาตรฐานเนื่องจากโปสเตอร์ไม่ได้ทำ ทำงานบน 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";
}
ฉันได้รับผลลัพธ์ที่สอดคล้องกันโดยมีตัวเลข 1 ล้านข้ามการพยายามนับสิบครั้ง:
- ล่วงหน้า: 0.7 วินาที
- แผนที่ปิด: 3.4 วินาที
- แม็พชื่อฟังก์ชัน: 1.2 วินาที
หากว่าความเร็วในแผนที่ที่ไม่ชัดเจนนั้นเกิดจากการปิดอาจถูกประเมินทุกครั้งฉันก็ทำการทดสอบเช่นนี้:
function useMapClosure($numbers) {
$closure = function($number) {
return $number * 10;
};
return array_map($closure, $numbers);
}
แต่ผลลัพธ์จะเหมือนกันโดยยืนยันว่าการปิดนั้นได้รับการประเมินเพียงครั้งเดียว
2014-02-02 UPDATE: opcodes dump
นี่คือ opcode ที่ทิ้งไว้สำหรับการโทรกลับทั้งสามครั้ง ครั้งแรก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