Bal*_*Fan 13 php variables foreach loops naming-conventions
那是真的真的,我可以用$_
一个虚拟变量在foreach
循环,如果没有必要$value
的foreach($array as $key => $value)
?除了PHP语法格式化之外,我找不到任何证明这一点的有用信息.
当循环内部没有使用该值时,foreach循环有一个特殊情况.在这种情况下,使用虚拟变量$ _(下划线):
foreach ($GLOBALS['TCA'] as $table => $_) { // Do something with $table }
这是出于性能原因而完成的,因为它比调用array_keys()并循环其结果更快.
LSe*_*rni 17
"_"是一个有效的变量名称字符,因此您可以像使用任何其他变量一样使用它,并且没有特殊意义; 这不是Perl.
<?php
$_ = "Hello";
$__ = "World";
$___ = "foo";
print "{$_}, {$__}, {$___}\n";
?>
Run Code Online (Sandbox Code Playgroud)
将按预期输出"Hello,World,foo".也,
foreach ( [ 'a' => 'Alpha', 'b' => 'Beta', 'c' => 'Gamma' ] as $letter => $_ ) {
print $letter;
}
print $_;
Run Code Online (Sandbox Code Playgroud)
将输出"abcGamma",表明该$_
变量在被用于之后仍保持定义foreach
; 它不是一种奇怪的"本地"变量.
至于表演,我觉得它没什么区别,但那是你的号召.相反,我会尝试不使用全局变量,以避免污染全局范围.
我认为,最近需要PHP
随时纠正/添加/建议改进
define('INNER_LOOP', 10000);
define('OUTER_LOOP', 10);
$TCA = [
'customers' => '',
'relations' => '',
'invoices' => '',
'books' => '',
'parts' => '',
'records' => '',
'calories' => '',
'bounties' => '',
'cats' => '',
'cowabunga' => '',
'amenities' => '',
];
$tests = [
"foreach access to global" => function() {
global $TCA;
for ($i = 0; $i < INNER_LOOP; $i++) {
foreach ($TCA as $table => $_) {
$t = $table . 'x';
}
}
},
"foreach access to GLOBALS" => function() {
for ($i = 0; $i < INNER_LOOP; $i++) {
foreach ($GLOBALS['TCA'] AS $table => $_) {
$t = $table . 'x';
}
}
},
"passing parameter" => function($TCA) {
for ($i = 0; $i < INNER_LOOP; $i++) {
foreach ($TCA AS $table => $_) {
$t = $table . 'x';
}
}
},
"passing parameter and array_keys" => function($TCA) {
$keys = array_keys($TCA);
for ($i = 0; $i < INNER_LOOP; $i++) {
foreach ($keys AS $table) {
$t = $table . 'x';
}
}
},
"walking passed parameter w/lambda" => function($TCA) {
for ($i = 0; $i < INNER_LOOP; $i++) {
array_map(
function($table) {
$t = $table . 'x';
},
array_keys($TCA)
);
}
},
"walking passed parameter w/ anon func" => function($TCA) {
$handler = function($table) {
$t = $table . 'x';
};
$keys = array_keys($TCA);
for ($i = 0; $i < INNER_LOOP; $i++) {
array_map($handler, $keys);
}
},
];
function timeFunc($function, $obj) {
$time = microtime(true);
for ($i = 0; $i < OUTER_LOOP; $i++) {
$function($obj);
}
return (microtime(true) - $time);
}
foreach ($tests as $name => $test) {
print "$name: " . timeFunc($test, $TCA) . "\n";
flush();
}
Run Code Online (Sandbox Code Playgroud)
这些是我的结果,格式化和排序:
- passing parameter and array_keys: 0.04573917388916
- foreach access to global: 0.067629098892212
- passing parameter: 0.08098292350769
- foreach access to GLOBALS: 0.082289934158325
- walking passed parameter w/ anon func: 1.6233508586884
- walking passed parameter w/lambda: 1.6796138286591
Run Code Online (Sandbox Code Playgroud)
有两点需要注意:在最快和最慢之间我有大约四十倍的差异.但是超过十万次通话的差异是1.63秒,这意味着在较快和较慢版本之间进行单次通话的时间为16.3 微秒.
因此,如果这些版本中的一个显示出拯救你的承诺,比如每年五分钟的头痛,寻找漏洞或客户支持,那么该版本的推出可能是值得投资的.
另一方面,如果你真的需要几十亿次的东西,那么那些微不足道的微秒加起来值得一试,那么你可能最好花一些时间来移植(或移植)那段代码一种语言本身更快或可以大规模并行化 - 可能是C或Erlang; 或重新思考架构(例如守护进程的过程,以节省开销,使用存储过程来卸载麻烦的RDBMS,缓存结果,...).