值得注意的是,`array_walk` 不能用于更改数组中的键。
该函数可以定义为 (&$value, $key),但不能定义为 (&$value, &$key)。
即使 PHP 没有报错/警告,它也不会修改键。
(PHP 4, PHP 5, PHP 7, PHP 8)
array_walk — 将用户提供的函数应用于数组的每个成员
将用户定义的 callback
函数应用于 array
数组的每个元素。
array_walk() 不受 array
的内部数组指针的影响。 array_walk() 将遍历整个数组,无论指针位置如何。
array
输入数组。
callback
通常,callback
接收两个参数。 array
参数的值作为第一个参数,键/索引作为第二个参数。
注意:
如果
callback
需要使用数组的实际值,请将callback
的第一个参数指定为 引用。 然后,对这些元素所做的任何更改都将在原始数组本身中进行。
注意:
许多内部函数(例如 strtolower())如果传递了超出预期数量的参数,则会发出警告,并且不能直接用作
callback
。
只有 array
的值可能会被更改;其结构不能被更改,即程序员不能添加、取消设置或重新排序元素。如果回调不遵守此要求,则此函数的行为未定义且不可预测。
arg
如果提供了可选的 arg
参数,它将作为第三个参数传递给 callback
。
始终返回 true
。
从 PHP 7.1.0 开始,如果 callback
函数需要超过 2 个参数(数组成员的值和键),或者如果也传递了 arg
则需要超过 3 个参数,则将抛出 ArgumentCountError。 以前,在这种情况下,每次 array_walk() 调用 callback
时都会生成 E_WARNING 级别错误。
示例 #1 array_walk() 示例
<?php
$fruits = array("d" => "lemon", "a" => "orange", "b" => "banana", "c" => "apple");
function test_alter(&$item1, $key, $prefix)
{
$item1 = "$prefix: $item1";
}
function test_print($item2, $key)
{
echo "$key. $item2\n";
}
echo "Before ...:\n";
array_walk($fruits, 'test_print');
array_walk($fruits, 'test_alter', 'fruit');
echo "... and after:\n";
array_walk($fruits, 'test_print');
?>
以上示例将输出
Before ...: d. lemon a. orange b. banana c. apple ... and after: d. fruit: lemon a. fruit: orange b. fruit: banana c. fruit: apple
示例 #2 使用匿名函数的 array_walk() 示例
<?php
$elements = ['a', 'b', 'c'];
array_walk($elements, function ($value, $key) {
echo "{$key} => {$value}\n";
});
?>
以上示例将输出
0 => a 1 => b 2 => c
值得注意的是,`array_walk` 不能用于更改数组中的键。
该函数可以定义为 (&$value, $key),但不能定义为 (&$value, &$key)。
即使 PHP 没有报错/警告,它也不会修改键。
在类中调用数组遍历
如果类是静态的
array_walk($array, array('self', 'walkFunction'));
或者
array_walk($array, array('className', 'walkFunction'));
否则
array_walk($array, array($this, 'walkFunction'));
我注意到
PHP 在使用 `array_walk()` 时忽略了参数类型,即使存在
declare(strict_types=1)
。
以下代码为例...
<?php
declare(strict_types=1);
$fruits = array("butter" => 5.3, "meat" => 7, "banana" => 3);
function test_print(int $item2, $key) {
echo "$key: $item2<br />\n";
}
array_walk($fruits, 'test_print');
?>
输出结果是
butter: 5
meat: 7
banana: 3
而期望的输出结果是
致命错误:未捕获的类型错误:传递给 test_print() 的参数 1 必须是整数类型
因为 "butter" => 5.3 是浮点数
我问了别人,他们说“这是因为从内部代码调用的回调函数将始终使用弱类型”。但我尝试了一些测试,在使用 `call_user_func()` 时,此行为不是问题。
针对 'ibolmo' 的回复,这是 `string_walk` 的扩展版本,允许传递用户数据(如 `array_walk`)并以与 `array_walk` 允许的方式编辑字符串,但现在请注意,您必须传递一个变量,因为 PHP 不能通过引用传递字符串字面量(逻辑上)。
<?php
function string_walk(&$string, $funcname, $userdata = null) {
for($i = 0; $i < strlen($string); $i++) {
# NOTE: PHP's dereference sucks, we have to do this.
$hack = $string{$i};
call_user_func($funcname, &$hack, $i, $userdata);
$string{$i} = $hack;
}
}
function yourFunc($value, $position) {
echo $value . ' ';
}
function yourOtherFunc(&$value, $position) {
$value = str_rot13($value);
}
# NOTE: We now need this ugly $x = hack.
string_walk($x = 'interesting', 'yourFunc');
// Ouput: i n t e r e s t i n g
string_walk($x = 'interesting', 'yourOtherFunc');
echo $x;
// Output: vagrerfgvat
?>
另请注意,直接在 `$x` 上调用 `str_rot13()` 会快得多 ;-) 只是一个示例。
请注意,将 `array_walk` 与 `intval` 一起使用是不合适的。
互联网上有很多示例建议使用以下代码来安全地转义 `$_POST` 整数数组
<?php
array_walk($_POST['something'],'intval'); // 在 PHP 5.3.3 中不起作用
?>
它在一些旧版本的 PHP(5.2)中有效,但违反了规范。由于 `intval()` 不会修改其参数,而是返回修改后的结果,因此以上代码对数组没有影响,并且会留下网站的安全漏洞。
您可以改为使用以下代码
<?php
$_POST['something'] = array_map(intval,$_POST['something']);
?>
大约 3 年前有一条关于将其用于修剪的说明。`array_map()` 对于此目的可能更简洁。我还没有检查时间/资源的影响
$result = array_map("trim", $array);
不要忘记 `array_map()` 函数,它可能更容易使用!
以下是如何将数组中的所有元素转换为小写
<?php
$arr = array_map('strtolower', $arr);
?>
由于 `array_walk` 不能修改/更改/重新索引键(如前所述),我提供了一个小的包装函数,它使用闭包和“use”关键字实现了传递数组引用和索引。
function indexArrayByElement($array, $element)
{
$arrayReindexed = [];
array_walk(
$array,
function ($item, $key) use (&$arrayReindexed, $element) {
$arrayReindexed[$item[$element]] = $item;
}
);
return $arrayReindexed;
}
我想遍历一个数组并将其反向映射到第二个数组中。我决定使用 `array_walk`,因为它应该比重置、下一个循环或 `foreach(x as &$y)` 循环更快。
<?php
$output = array();
array_walk($input, 'gmapmark_reverse', $output);
function gmapmark_reverse(&$item, $index, &$target) {
$target[$item['form_key']] = $index;
}
?>
在我的调试器中,我可以看到 `$target` 正在逐步更新,但当 `array_walk` 返回时,`$output` 为空。但是,如果我使用(已弃用)的按引用传递
<?php
array_walk($input, 'gmapmark_reverse', &$output);
?>
`$output` 将正确返回。不幸的是,没有简单的方法可以抑制警告
<?php
@array_walk($input, 'gmapmark_reverse', &$output);
?>
不会使它们静音。我设计了一种使用静态数组的解决方法
<?php
$reverse = array();
array_walk($input, 'gmapmark_reverse');
// 最后一次调用函数以获取目标数组,因为参数不起作用
$reverse = gmapmark_reverse($reverse);
function gmapmark_reverse(&$item, $index = 0) {
static $target;
if (!$target) {
$target = array();
}
if (isset($item['form_key'])) {
$target[$item['form_key']] = $index;
}
return($target);
}
?>
在回调函数中永久修改第三个(可选)参数时,通过引用传递它非常有用。这会导致将修改后的参数传递给 `array_walk()` 的下一次迭代。下面的示例枚举了数组中的项目
<?php
function enumerate( &$item1, $key, &$startNum ) {
$item1 = $startNum++ ." $item1";
}
$num = 1;
$fruits = array( "lemon", "orange", "banana", "apple");
array_walk($fruits, 'enumerate', $num );
print_r( $fruits );
echo '$num is: '. $num ."\n";
?>
输出结果如下
数组
(
[0] => 1 lemon
[1] => 2 orange
[2] => 3 banana
[3] => 4 apple
)
$num is: 1
注意输出的最后一行,在 `array_walk()` 函数外部,`$num` 参数的值仍然是初始值 1。这是因为 `array_walk()` 函数的第三个参数不是按引用传递的。那么,如果我们将引用作为可选参数传递会怎样呢?
<?php
$num = 1;
$fruits = array( "lemon", "orange", "banana", "apple");
array_walk($fruits, 'enumerate', &$num ); // 这里传递引用
print_r( $fruits );
echo '$num is: '. $num ."\n";
echo "我们有 ". ($num - 1) ." 个水果在篮子里!";
?>
输出结果如下
数组
(
[0] => 1 lemon
[1] => 2 orange
[2] => 3 banana
[3] => 4 apple
)
$num is: 5
我们有 4 个水果在篮子里!
现在 `$num` 的值发生了改变,因此我们可以计算项目的数量(无需不必要地调用 `count()` 函数)。
总之,在 `array_walk()` 函数中使用引用可以是一种强大的工具,但应该谨慎使用,因为在 `array_walk()` 函数外部修改第三个参数并不总是我们想要的。
// 我们可以使用简单的 FOREACH 循环来实现这一点
$fruits = array("d" => "lemon", "a" => "orange", "b" => "banana", "c" => "apple");
foreach($fruits as $cls => $vls)
{
$fruits[$cls] = "fruit: ".$vls;
}
结果
数组
(
[d] => fruit: lemon
[a] => fruit: orange
[b] => fruit: banana
[c] => fruit: apple
)
对于那些认为无法使用 `array_walk` 来更改/替换键名的人,请看这里
<?php
function array_explore(array &$array, callable $callback)
{
array_walk($array, function(&$value, $key) use (&$array, $callback)
{
$callback($array, $key, $value);
if(is_array($value))
{
array_explore($value, $callback);
}
});
}
/**
* 摘自:https://stackoverflow.com/questions/13233405/change-key-in-associative-array-in-php
*/
function renameKey(array &$data, $oldKey, $newKey, $ignoreMissing = false, $replaceExisting = false)
{
if (!empty($data))
{
if (!array_key_exists($oldKey, $data))
{
if ($ignoreMissing)
{
return FALSE;
}
throw new \Exception('旧键不存在。');
}
else
{
if (array_key_exists($newKey, $data))
{
if ($replaceExisting)
{
unset($data[$newKey]);
}
else
{
throw new \Exception('新键已存在。');
}
}
$keys = array_keys($data);
// 来自 EllisGL 的修复:https://php.net/manual/en/function.array-search.php#122377
$keys[array_search($oldKey, array_map('strval', $keys))] = $newKey;
$data = array_combine($keys, $data);
return TRUE;
}
}
return FALSE;
}
$array = [
"_10fish" => 'xyz',
"_11fish" => [
"_22" => "a", "b", "c"
],
"someFish" => [
'xyz',
'@attributes' => ['type' => 'cod']
]
];
array_explore($array, function(&$value, $key)
{
// 将键 '@attrutes' 替换为 '_attributes'
if('@attributes' === $key)
{
renameKey($value, $key, '_attributes');
}
});
print_r($array);
?>
使用 lambda 表达式,可以创建一个方便的 zip 函数来将数组的键和值组合在一起。我扩展了它,允许您将“粘合”字符串作为可选的用户数据参数传递。以下示例用于组合电子邮件标题的数组
<?php
/**
* 使用提供的粘合符将数组的键和值组合在一起
*
* 数组的值将替换为新的计算值
*
* @param array $data
* @param string $glue
*/
function zip(&$data, $glue=': ')
{
if(!is_array($data)) {
throw new InvalidArgumentException('第一个参数必须是数组');
}
array_walk($data, function(&$value, $key, $joinUsing) {
$value = $key . $joinUsing . $value;
}, $glue);
}
$myName = 'Matthew Purdon';
$myEmail = '[email protected]';
$from = "$myName <$myEmail>";
$headers['From'] = $from;
$headers['Reply-To'] = $from;
$headers['Return-path'] = "<$myEmail>";
$headers['X-Mailer'] = "PHP" . phpversion() . "";
$headers['Content-Type'] = 'text/plain; charset="UTF-8"';
zip($headers);
$headers = implode("\n", $headers);
$headers .= "\n";
echo $headers;
/*
From: Matthew Purdon <[email protected]>
Reply-To: Matthew Purdon <[email protected]>
Return-path: <[email protected]>
X-Mailer: PHP5.3.2
Content-Type: text/plain; charset="UTF-8"
*/
?>
`array_walk` 函数不适用于 `SplFixedArray` 对象
<?php
$array = new SplFixedArray(2);
$array[0] = 'test_1';
$array[1] = 'test_2';
array_walk($array, function(&$val){
$val .= '__';
return $val;
});
foreach ($array as $a) {
echo "$a\n";
}
?>
结果是
test_1__
test_2__
一些速度测试
<?php
// 测试结果
$array1 = test('array_walk');
$array2 = test('array_walk_list_each');
$array3 = test('array_walk_foreach1');
$array4 = test('array_walk_foreach2');
// 检查数组是否相等
var_dump($array1 == $array2, $array1 == $array3, $array1 == $array4);
// 测试函数 1
function array_walk_list_each(&$array, $function, $userData = null) {
while ( list($key, $value) = each($array) )
$function($array[$key], $key, $userData);
}
// 测试函数 2
function array_walk_foreach1(&$array, $function, $userData = null) {
foreach ($array as $key => &$value )
$function($value, $key, $userData);
}
// 测试函数 3
function array_walk_foreach2(&$array, $function, $userData = null) {
foreach ($array as $key => $value )
$function($array[$key], $key, $userData);
}
function some_function(&$value, $key, $userData) {
$value = "$key => $userData";
}
function test($function, $count = 10000, $arrayElements = 1000) {
echo $function, ' ... ';
$array = array_fill(0, $arrayElements, "some text value");
$timer = microtime(true);
for( $i = 0; ++$i < $count; )
$function($array, 'some_function', 'some user data');
printf("%.3f sec\n", microtime(true) - $timer);
return $array;
}
?>
输出 (PHP 5.4.9-4ubuntu2.2 (cli) (built: Jul 15 2013 18:24:39))
=========================
array_walk ... 13.572 sec
array_walk_list_each ... 0.027 sec
array_walk_foreach1 ... 15.356 sec
array_walk_foreach2 ... 17.416 sec
bool(true)
bool(true)
bool(true)
输出 (PHP 5.5.0 (cli) (built: Jul 16 2013 17:59:42) - 同一台服务器)
=========================
array_walk ... 4.776 sec
array_walk_list_each ... 0.006 sec
array_walk_foreach1 ... 4.482 sec
array_walk_foreach2 ... 5.166 sec
bool(true)
bool(true)
bool(true)
PHP 5.5 中 array_walk 看起来相当不错,但是 list each 越来越快了...
对 zlobnygrif 的速度测试的更正。
<?php
// 测试结果
$array1 = test('array_walk');
$array2 = test('array_walk_list_each');
$array3 = test('array_walk_foreach1');
$array4 = test('array_walk_foreach2');
// 检查数组是否相等
var_dump($array1 == $array2, $array1 == $array3, $array1 == $array4);
// 测试函数 1
function array_walk_list_each(&$array, $function, $userData = null) {
/* 确保每次都遍历数组 */
reset($array);
while ( list($key, $value) = each($array) )
$function($array[$key], $key, $userData);
}
// 测试函数 2
function array_walk_foreach1(&$array, $function, $userData = null) {
foreach ($array as $key => &$value )
$function($value, $key, $userData);
}
// 测试函数 3
function array_walk_foreach2(&$array, $function, $userData = null) {
foreach ($array as $key => $value )
$function($array[$key], $key, $userData);
}
function some_function(&$value, $key, $userData) {
$value = "$key => $userData";
}
function test($function, $count = 10000, $arrayElements = 1000) {
echo $function, ' ... ';
$array = array_fill(0, $arrayElements, "some text value");
$timer = microtime(true);
for( $i = 0; ++$i < $count; )
/* 每次更改数据 */
$function($array, 'some_function', 'some user data ' . $i);
printf("%.3f sec\n", microtime(true) - $timer);
return $array;
}
不幸的是,我花了大量时间尝试使用 array_walk 函数永久应用函数对数组的影响,而实际上我想要的是 array_map。对于那些可能对这个函数感到非常沮丧的人,这里有一个非常简单但有效的例子...
<?php
$fruits = array("Lemony & Fresh","Orange Twist","Apple Juice");
print_r($fruits);
echo '<br />';
function name_base($key)
{
$name2 = str_replace(" ", "_", $key);
$name3 = str_replace("&", "and", $name2);
$name4 = strtolower($name3);
echo $name4.'<br />';
return $name4;
}
echo '<br />';
$test = array_map('name_base', $fruits);
$fruits_fixed = $test;
echo '<br />';
print_r($fruits_fixed);
?>
为了完整起见,必须提及在 PHP 5.3 中使用闭包函数的可能性。
<?php
$names = array("D\'Artagnan", "Aramis", "Portos");
array_walk($names, function(&$n) {
$n = stripslashes($n);
});
?>
使用 `array_walk` 的陷阱在于它不返回数组,而是通过引用修改它。
您可以使用 Lambda 函数作为第二个参数。
<?php
array_walk($myArray, function(&$value, $key){
// 如果您想更改数组值,则在 $value 之前使用 "&" 是必须的。
});
?>
示例(将正值乘以二)
<?php
$myArray = array(1, 2, 3, 4, 5);
array_walk($myArray, function(&$value, $index){
if ($value > 0) $value *= 2;
});
?>
使用闭包的示例,检查和删除数组中的值
<?php
$array = array('foo' => 'bar', 'baz' => 'bat');
array_walk($array, function($val,$key) use(&$array){
if ($val == 'bar') {
unset($array[$key]);
}
});
var_dump($array);
在数组值前面加上键并将其作为粘贴字符串检索,原始数组保持不变。我用它从数组创建一些 SQL 查询。
<?php
function array_implode_prefix($outer_glue, $arr, $inner_glue, $prefix=false){
array_walk( $arr , "prefix", array($inner_glue, $prefix) );
return implode($outer_glue, $arr);
}
function prefix(&$value, $key, array $additional){
$inner_glue = $additional[0];
$prefix = isset($additional[1])? $additional[1] : false;
if($prefix === false) $prefix = $key;
$value = $prefix.$inner_glue.$value;
}
//示例 1:
$order_by = array("3"=>"ASC", "2"=>"DESC", "7"=>"ASC");
echo array_implode_prefix(",", $order_by, " ");
//输出:3 ASC,2 DESC,7 ASC
//示例 2:
$columns = array("product_id", "category_id", "name", "description");
$table = "product";
echo array_implode_prefix(", ", $columns, ".", $table);
//输出:product.product_id, product.category_id, product.name, product.description
//示例 3 (prefix 函数) 实际上不会单独使用
$pre= "vacation";
$value = "lalaland";
prefix($value, $pre, array("."));
echo $value;
//输出:vacation.lalaland
?>
此函数可用于将一个数组中的值添加到另一个数组中。
<?php
echo '<pre>';
// 示例代码
$a = [ 1, 2, 3, 4, 5, 6];
$c = [10,20,30,40,50,60];
array_walk($a,
function(&$item, $key, $y) { $item += $y[$key];},
$c
);
print_r($a);
// 结果
Array
(
[0] => 11
[1] => 22
[2] => 33
[3] => 44
[4] => 55
[5] => 66
)
我一直在寻找修剪数组中所有元素的方法,我发现这是最简单的解决方案。
<?php
array_walk($ids, create_function('&$val', '$val = trim($val);'));
?>
public function big_endian_array_walk(array $array, $callback) {
end($array);
for($i=sizeof($array);$i>0;$i--) {
$key = key($array);
$value = array_pop($array);
if(preg_match('/^[a-zA-Z_\x80-\xff][a-zA-Z0-9_\x80-\xff]*$/', $value)) {
call_user_func_array($callback, [$value, $key]);
}
}
}
我只是想从后往前遍历。
public function big_endian_array_walk(array $array, $callback) {
end($array);
for($i=sizeof($array);$i>0;$i--) {
$key = key($array);
$value = array_pop($array);
if(preg_match('/^[a-zA-Z_\x80-\xff][a-zA-Z0-9_\x80-\xff]*$/', $value)) {
call_user_func_array($callback, [$value, $key]);
}
}
}
只有通过网页浏览器查看时,示例的输出才是正确的。但是,如果您将其传递给 PHP-CLI,则会看到额外的 HTML 换行符。
当我在 php5.2.5 中通过引用传递第三个参数时,
发生了这种情况:警告:调用时按引用传递已弃用 - 参数按值传递...
并且根据http://bugs.php.net/bug.php?id=19699,在 php.ini 中将 allow_call_time_pass_reference 设置为 true 不会起作用,因此要解决此问题
<?php
array_walk($arrChnOut, create_function('&$v, $k, $arr_rtn', 'if ($k{0}!="_") {$arr_rtn[0]["_".$v[\'ID\']]=$v; unset($arr_rtn[0][$k]);}'), array(&$arrChnOut));
?>
对于所有试图将 trim() 硬塞进 array_walk() 并找到所有这些技巧来解决 array_walk() 向回调传递 2 个参数的问题的人...
看看 array_map()。
https://php.net/array_map
它非常棒。
顺便说一下,我也是这些人之一,在 15 年的 php 开发之后,我很高兴地说我仍在学习新东西。:) 我自己刚刚了解了 array_map()...
这是一个简单且易于使用的此函数的实现。
“原始”函数存在无法取消设置值的问题。
使用我的函数,您可以取消设置!
<?php
function array_walk_protected(&$a,$s,$p=null)
{
if(!function_exists($s)||!is_array($a))
{
return false;
}
foreach($a as $k=>$v)
{
if(call_user_func_array($s,array(&$a[$k],$k,$p))===false)
{
unset($a[$k]);
}
}
}
function get_name(&$e,$i,$p)
{
echo "$i: $e<br>";
return false;
}
$m=array('d'=>'33','Y'=>55);
array_walk_protected($m,'get_name');
var_dump($m); //返回数组(0) { }
?>
我称之为 array_walk_protected,因为它可以防止使用原始函数取消设置值时出现的意外行为。
要删除元素,只需返回 false 即可!
无需其他操作!
在您创建的函数中取消设置 $e 将使数组保持不变,没有任何更改!
顺便说一句,如果 $a 不是数组或 $s 不是字符串,则该函数将返回 false!
限制:它只能运行用户定义的函数。
希望您喜欢它!
您好,
如果要将两个数组中具有相同键的值相加
<?php
function add(&$item,$key,$search) {
$item += (is_array($search))?((isset($search[$key]))?$search[$key]:0):0;
}
$a = ["orange" => 2, "banana" => 3, "apple" => 1];
$b = ["orange" => 1, "apple" => 4];
array_walk($c,"add",$b);
echo "<pre>".print_r($c,true)."</pre>";
?>
这将输出
"orange" => 3,
"banana" => 3,
"apple" => 5
如果您想修改多维数组的每个值,请使用此处使用的此函数
<?php
$array = array (1=>1, 2=> 2, 3 => array(1=>11, 2=>12, 3=>13));
$text = "test";
function modarr(&$array, $text) {
foreach ($array as $key => $arr) {
if(is_array($arr)) $res[$key] = modarr(&$arr,$text);
//修改函数在此处
else $res[$key] = $arr.$text;
}
return $res;
}
$erg = modarr($array, $text);
print_r($erg);
?>
结果将是
<?php
Array ( [1] => 1test [2] => 2test [3] => Array ( [1] => 11test [2] => 12test [3] => 13test ) )
?>
您想删除用户在表单字段中添加的空格吗?
只需使用...
类 SomeVeryImportantClass
{
...
public function mungeFormData(&$data)
{
array_walk($data, array($this, 'munge'));
}
private function munge(&$value, &$key)
{
if(is_array($value))
{
$this->mungeFormData($value);
}
else
{
$value = trim($value);
}
}
...
}
所以...
$obj = new SomeVeryImportantClass;
$obj->mungeFormData($_POST);
___
eNc
如果您在函数内部使用全局的临时返回数组,则可以使用 array_walk 更改键或值。例如
$array = ['a'=>10, 'b'=>20];
$sequence = array ();
$newArray = array_values(array_walk($array, 'fn'));
function fn(&$val,$key){
global $sequence;
$sequence [] = $val;
}
无需担心子数组的内部指针位置。您现在已倒带,拥有一个基于 0 的新数组,而不是字符串键。
我在使用此函数时遇到了一些问题 - 它不想应用 PHP 定义的函数。所以我决定自己写一个 - 这里就是。我不得不使用一些泛型编程技能,并没有真正检查速度(我认为它可能很慢)...我相信它可以做得更好,但我不知道如何 - 好吧,我想多数组支持和递归会很好。所以呢?
原型
bool arrayWalk(array &$arry, callback $callback, mixed $params=false)
<?php
function arrayWalk(&$arry, $callback, $params=false) {
$P=array(""); // 参数
$a=""; // 参数字符串 :)
if($params !== false) { // 添加参数
if(is_array($params)) { // 多个附加参数
foreach($params as $par)
{ $P[]=$par; }
}
else // 只有一个附加参数
{ $P[]=$params; }
}
for( // 创建参数字符串
$i=0; isset($P[$i]); ++$i
)
{ $a.='$'.chr($i + 97).', '; } // 随机参数名称
$a=substr($a, 0, -2); // 去除最后一个逗号和两个空格
$func=create_function($a, 'return '.$callback.'('.$a.');'); // 通用函数
if(is_callable($func)) {
for( // 遍历数组
$i=0; isset($arry[$i]); ++$i
) {
$P[0]=$arry[$i]; // 第一个元素必须是第一个参数 - 数组值
$arry[$i] = call_user_func_array($func, $P); // 使用通用函数获得新值并赋值
}
}
else
{ return false; } // 失败 - 函数不可调用
return true; // 成功!
} // arrayWalk()
?>
到目前为止,我注意到一个大问题 - 例如,如果你想对数组使用 str_replace,你会失败 - 仅仅是因为 str_replace 的参数顺序,其中修改的字符串是第三个参数,而不是 arrayWalk 所需的第一个参数。
所以,还有一些工作要做……