回调如何用PHP编写?
该手册可互换地使用术语"回调"和"可调用",但是,"回调"传统上是指充当函数指针的字符串或数组值,引用函数或类方法以供将来调用。自PHP 4以来,已经允许使用函数式编程的某些元素。
1 2 3 4 5 6 7 8 9 10 11
| $cb1 = 'someGlobalFunction';
$cb2 = ['ClassName', 'someStaticMethod'];
$cb3 = [$object, 'somePublicMethod'];
// this syntax is callable since PHP 5.2.3 but a string containing it
// cannot be called directly
$cb2 = 'ClassName::someStaticMethod';
$cb2(); // fatal error
// legacy syntax for PHP 4
$cb3 = array(&$object, 'somePublicMethod'); |
通常,这是使用可调用值的安全方法:
1 2 3 4 5 6 7 8
| if (is_callable($cb2)) {
// Autoloading will be invoked to load the class"ClassName" if it's not
// yet defined, and PHP will check that the class has a method
//"someStaticMethod". Note that is_callable() will NOT verify that the
// method can safely be executed in static context.
$returnValue = call_user_func($cb2, $arg1, $arg2);
} |
现代PHP版本允许上面的前三种格式直接作为$cb()调用。 call_user_func和call_user_func_array支持以上所有内容。
参见:http://php.net/manual/en/language.types.callable.php
注释/注意事项:
如果函数/类已命名空间,则字符串必须包含标准名称。例如。 ['Vendor\Package\Foo', 'method']
call_user_func不支持通过引用传递非对象,因此您可以使用call_user_func_array,或者在更高的PHP版本中,将回调保存到var并使用直接语法:$cb();
具有__invoke()方法的对象(包括匿名函数)属于"可调用"类别,并且可以按相同方式使用,但是我个人不将其与旧式"回调"术语相关联。
旧版create_function()创建一个全局函数并返回其名称。它是eval()的包装,应改用匿名函数。
使用PHP 5.3,您现在可以执行以下操作:
1 2 3 4 5
| function doIt($callback) { $callback(); }
doIt(function() {
// this will be done
}); |
最后,这是一个不错的方法。对PHP的重要补充,因为回调很棒。
回调的实现是这样完成的
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| // This function uses a callback function.
function doIt($callback)
{
$data ="this is my data";
$callback($data);
}
// This is a sample callback function for doIt().
function myCallback($data)
{
print 'Data is: ' . $data . "
";
}
// Call doIt() and pass our sample callback function's name.
doIt('myCallback'); |
显示:数据是:这是我的数据
我最近发现的一个妙招是使用PHP的create_function()创建用于一次使用的匿名/ lambda函数。对于使用回调进行自定义处理的PHP函数(如array_map(),preg_replace_callback()或usort())很有用。看起来很像是在幕后做eval(),但这仍然是使用PHP的一种不错的功能风格方式。
您将要验证您的通话是否有效。例如,对于特定功能,您将需要检查并查看该功能是否存在:
1 2 3 4 5 6 7
| function doIt($callback) {
if(function_exists($callback)) {
$callback();
} else {
// some error handling
}
} |
好吧……随着5.3的出现,一切都会变得更好,因为有了5.3,我们将获得闭包,并获得匿名函数
http://wiki.php.net/rfc/closures
create_function在课堂上对我不起作用。我不得不使用call_user_func。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| <?php
class Dispatcher {
//Added explicit callback declaration.
var $callback;
public function Dispatcher( $callback ){
$this->callback = $callback;
}
public function asynchronous_method(){
//do asynch stuff, like fwrite...then, fire callback.
if ( isset( $this->callback ) ) {
if (function_exists( $this->callback )) call_user_func( $this->callback,"File done!" );
}
}
} |
然后,使用:
1 2 3 4 5 6 7 8 9 10
| <?php
include_once('Dispatcher.php');
$d = new Dispatcher( 'do_callback' );
$d->asynchronous_method();
function do_callback( $data ){
print 'Data is: ' . $data . "
";
}
?> |
[编辑]
添加了缺少的括号。
另外,添加了回调声明,我更喜欢这样。
每当我在php中使用create_function()时,我都会畏缩。
参数是一个用逗号分隔的字符串,整个函数体都在一个字符串中...哎呀...我认为即使尝试,它们也无法使其更难看。
不幸的是,创建命名函数不值得麻烦时,它是唯一的选择。
对于那些不想破坏与PHP < 5.4的兼容性的人,我建议使用类型提示进行更简洁的实现。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
| function call_with_hello_and_append_world( callable $callback )
{
// No need to check $closure because of the type hint
return $callback("hello" )."world";
}
function append_space( $string )
{
return $string."";
}
$output1 = call_with_hello_and_append_world( function( $string ) { return $string.""; } );
var_dump( $output1 ); // string(11)"hello world"
$output2 = call_with_hello_and_append_world("append_space" );
var_dump( $output2 ); // string(11)"hello world"
$old_lambda = create_function( '$string', 'return $string."";' );
$output3 = call_with_hello_and_append_world( $old_lambda );
var_dump( $output3 ); // string(11)"hello world" |