
在PHP中,当我们将一个方法调用的结果赋值给关联数组的元素时,PHP会立即执行该方法以获取其返回值,然后将这个返回值存储到数组中。这与我们期望的“存储一个待执行的方法”的初衷相悖。
考虑以下场景,我们试图将 ClassOne 中的多个任务方法作为值存储在 func_map 关联数组中,并期望它们在后续的过滤和遍历过程中才被执行:
// class_two.php 中存在的问题代码片段
class ClassTwo {
public function getValues(ClassOne &$class_one, array $filters){
$func_map = [
"task_1" => call_user_func_array(array($class_one, "task1"), array(1, 2)),
"task_2" => call_user_func_array(array($class_one, "task2"), array(1, 2, 3)),
"task_3" => call_user_func_array(array($class_one, "task3"), array(3))
];
// 这里的 array_intersect_key 旨在过滤,但方法已在此处全部执行
return array_intersect_key($func_map, array_flip($filters));
}
}当我们运行包含上述逻辑的代码时,即使 filters 数组中只包含 "task_1",task1、task2 和 task3 这三个方法也会在 getValues 方法被调用时立即全部执行。这是因为 call_user_func_array 函数本身就是一个立即执行的方法,它会立即调用其指定的函数或方法并返回结果。因此,关联数组中存储的是这些方法执行后的返回值,而不是方法本身或一个可执行的引用。
要实现将方法作为可执行实体存储并在需要时才调用,我们可以利用PHP的匿名函数(也称为闭包)。匿名函数可以将一段代码逻辑封装起来,并作为一个变量进行传递和存储,它本身并不会立即执行,而是在被显式调用时才执行其内部逻辑。
立即学习“PHP免费学习笔记(深入)”;
以下是使用匿名函数改进后的 getValues 方法:
// class_two.php 改进后的代码片段
class ClassTwo {
public function getValues(ClassOne &$class_one, array $filters){
$func_map = [
// 使用匿名函数封装方法调用
"task_1" => function() use ($class_one) { return $class_one->task1(1, 2); },
"task_2" => function() use ($class_one) { return $class_one->task2(1, 2, 3); },
"task_3" => function() use ($class_one) { return $class_one->task3(3); }
];
// 此时 $func_map 中存储的是匿名函数,而不是方法的返回值
return array_intersect_key($func_map, array_flip($filters));
}
}在这个改进后的代码中:
为了更全面地理解这一机制,我们来看一个完整的示例。
class_one.php (任务类定义)
<?php
class ClassOne {
public function __construct(){}
public function task1($param1, $param2){
echo "Performing task1 ..\n"; // 添加换行符以便观察
$value = $param1 + $param2;
echo "Result task1: {$value}\n";
return $value; // 返回计算值
}
public function task2($param1, $param2, $param3){
echo "Performing task2 ..\n";
return [$param1, $param2, $param3];
}
public function task3($param1){
echo "Performing task3 ..\n";
$result = [];
for($i = 0; $i < 3; $i++){ // 简化循环次数
$result[] = $param1 * $i;
}
return $result;
}
}
?>class_two.php (包含延迟执行逻辑的类)
<?php
class ClassTwo {
public function __construct(){}
public function getValues(ClassOne &$class_one, array $filters){
$func_map = [
"task_1" => function() use ($class_one) { return $class_one->task1(1, 2); },
"task_2" => function() use ($class_one) { return $class_one->task2(1, 2, 3); },
"task_3" => function() use ($class_one) { return $class_one->task3(3); }
];
// 根据过滤器返回需要执行的任务
return array_intersect_key($func_map, array_flip($filters));
}
}
?>index.php (主执行文件)
<html>
<head>
<title>PHP Test</title>
</head>
<body>
<?php
include("class_one.php");
include("class_two.php");
$class_one = new ClassOne();
$class_two = new ClassTwo();
// 假设我们只想执行 task_1
$filters = ["task_1"];
$func_map = $class_two->getValues($class_one, $filters);
echo "--- 开始执行过滤后的任务 ---\n";
foreach($func_map as $key => $func){
// 此时 $func 是一个匿名函数,通过 $func() 调用它
$result = $func();
echo "Task '{$key}' executed, result type: " . gettype($result) . "\n";
var_dump($result); // 打印任务的实际返回值
}
echo "--- 所有任务执行完毕 ---\n";
// 再次 var_dump $func_map,此时它只包含被过滤后的匿名函数
echo "--- 最终 func_map 内容 ---\n";
var_dump($func_map);
?>
</body>
</html>运行 index.php 后,您将观察到以下输出:
--- 开始执行过滤后的任务 ---
Performing task1 ..
Result task1: 3
Task 'task_1' executed, result type: integer
int(3)
--- 所有任务执行完毕 ---
--- 最终 func_map 内容 ---
array(1) {
["task_1"]=>
class Closure#3 (1) {
// ... 匿名函数的内部表示,通常包含use的变量和代码信息
}
}从输出中可以看出,只有 task1 被执行了。Performing task2 .. 和 Performing task3 .. 不再出现,这证明了我们通过匿名函数成功实现了方法的延迟执行。var_dump($func_map) 显示 task_1 的值是一个 Closure 对象,而不是 int(3),进一步证实了这一点。
在使用匿名函数实现延迟执行时,有几个重要的注意事项和最佳实践:
参数传递与捕获:
性能考量: 创建匿名函数并捕获变量会带来微小的额外开销。对于需要大量创建和存储匿名函数的场景,应评估其对性能的影响。然而,在大多数业务逻辑中,这种开销通常
以上就是PHP关联数组中延迟执行方法:使用匿名函数的详细内容,更多请关注php中文网其它相关文章!
PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号