php手册之Callback 回调类型

手册目录: 语言参考---类型---Callback回调类型

参考详情: https://secure.php.net/manual/zh/language.types.callable.php

评论部分:

1. by edanschwartz@gmail.com

你可以使用'self::methodName'作为一个回调函数,但是这样做是很危险的,参见一下例子:

<?php
class Foo {
    public static function
doAwesomeThings() {
       
FunctionCaller::callIt('self::someAwesomeMethod');
    }

    public static function
someAwesomeMethod() {
       
// fantastic code goes here.
   
}
}

class
FunctionCaller {
    public static function
callIt(callable $func) {
       
call_user_func($func);
    }
}

Foo::doAwesomeThings();

?>

运行出错:class 'FunctionCaller' does not have a method 'someAwesomeMethod'.因为FunctionCaller并不知道self对应着Foo

基于此,你应该始终使用全类名进行调用,如下:

<?php
FunctionCaller
::callIt('Foo::someAwesomeMethod');
?>

2. by computrius@gmail.com

当你指明类的方法以数组的形式回调时(例如.array($this,'myFunc')),回调的方法可以是私有的,但此种情况只适用于类内调用,如果类外调用私有方法将会报错.

<?php

class mc {
   public function
go(array $arr) {
      
array_walk($arr, array($this, "walkIt"));
   }

   private function
walkIt($val) {
       echo
$val . "<br />";
   }

    public function
export() {
        return array(
$this, 'walkIt');
    }
}

$data = array(1,2,3,4);

$m = new mc;
$m->go($data); // valid

array_walk($data, $m->export()); // 将会产生警告
?>

输出:1<br />2<br />3<br />4<br />

warning:array_walk() expects parameter 2 to be a valid callback, cannot access private method mc::walkIt() in /in/tfh7f on line 22.

3. by Riikka K

他认为使用callable和可变函数进行回调,可能会产生错误,如下代码:

<?php
class foo {
    static function
callIt(callable $callback) {
       
$callback();
    }
   
    static function
doStuff() {
        echo
"Hello World!";
    }
}
foo::callIt('foo::doStuff');
?>

会产生一个error: Fatal error: Call to undefined function foo::doStuff() in /tmp/code.php on line 4

但是经过我个人在php7上测试,完美通过,所以可能是Riikka K使用的php版本较低的缘故,这里说明一下.

4. by andrewbessa@gmail.com

你可以使用$this来指定一个回调函数,例如:

<?php
class MyClass {

    public
$property = 'Hello World!';

    public function
MyMethod()
    {
       
call_user_func(array($this, 'myCallbackMethod'));
    }

    public function
MyCallbackMethod()
    {
        echo
$this->property;
    }

}
?>

5. by metemarkers@gmail.com

你可以像回调一个方法那样回调一个实现了__invoke()魔术方法的对象,__invoke()方法会在你尝试以调用函数的形式调用对象时被自动调用

metemarkers没有给出具体的实例,这里添加一个我自己的例子:

<?php
class CallableClass 
{
    function 
__invoke($x) {
        
var_dump($x);
    }
}
$obj = new CallableClass;
$obj(5);   //以函数的形式进行调用,会调用__invoke(),输出int(5)
var_dump(is_callable($obj));   //是callable类型的
?>
输出: int(5) bool(true)

6. by mariano.REMOVE.perez.rodringuez@gmail.com

这哥们写了一个方法来判定传入的方法是否是callable类型,并可以同时作用多种回调方式,如下是源码:

<?php
/**
* The callable types and normalizations are given in the table below:
*
*  Callable                        | Normalization                   | Type
* ---------------------------------+---------------------------------+--------------
*  function (...) use (...) {...}  | function (...) use (...) {...}  | 'closure'
*  $object                         | $object                         | 'invocable'
*  "function"                      | "function"                      | 'function'
*  "class::method"                 | ["class", "method"]             | 'static'
*  ["class", "parent::method"]     | ["parent of class", "method"]   | 'static'
*  ["class", "self::method"]       | ["class", "method"]             | 'static'
*  ["class", "method"]             | ["class", "method"]             | 'static'
*  [$object, "parent::method"]     | [$object, "parent::method"]     | 'object'
*  [$object, "self::method"]       | [$object, "method"]             | 'object'
*  [$object, "method"]             | [$object, "method"]             | 'object'
* ---------------------------------+---------------------------------+--------------
*  other callable                  | idem                            | 'unknown'
* ---------------------------------+---------------------------------+--------------
*  not a callable                  | null                            | false
*
* If the "strict" parameter is set to true, additional checks are
* performed, in particular:
*  - when a callable string of the form "class::method" or a callable array
*    of the form ["class", "method"] is given, the method must be a static one,
*  - when a callable array of the form [$object, "method"] is given, the
*    method must be a non-static one.
*
*/
function callableType($callable, $strict = true, callable& $norm = null) {
  if (!
is_callable($callable)) {
    switch (
true) {
      case
is_object($callable):
       
$norm = $callable;
        return
'Closure' === get_class($callable) ? 'closure' : 'invocable';
      case
is_string($callable):
       
$m    = null;
        if (
preg_match('~^(?<class>[a-z_][a-z0-9_]*)::(?<method>[a-z_][a-z0-9_]*)$~i', $callable, $m)) {
          list(
$left, $right) = [$m['class'], $m['method']];
          if (!
$strict || (new \ReflectionMethod($left, $right))->isStatic()) {
           
$norm = [$left, $right];
            return
'static';
          }
        } else {
         
$norm = $callable;
          return
'function';
        }
        break;
      case
is_array($callable):
       
$m = null;
        if (
preg_match('~^(:?(?<reference>self|parent)::)?(?<method>[a-z_][a-z0-9_]*)$~i', $callable[1], $m)) {
          if (
is_string($callable[0])) {
            if (
'parent' === strtolower($m['reference'])) {
              list(
$left, $right) = [get_parent_class($callable[0]), $m['method']];
            } else {
              list(
$left, $right) = [$callable[0], $m['method']];
            }
            if (!
$strict || (new \ReflectionMethod($left, $right))->isStatic()) {
             
$norm = [$left, $right];
              return
'static';
            }
          } else {
            if (
'self' === strtolower($m['reference'])) {
              list(
$left, $right) = [$callable[0], $m['method']];
            } else {
              list(
$left, $right) = $callable;
            }
            if (!
$strict || !(new \ReflectionMethod($left, $right))->isStatic()) {
             
$norm = [$left, $right];
              return
'object';
            }
          }
        }
        break;
    }
   
$norm = $callable;
    return
'unknown';
  }
 
$norm = null;
  return
false;
}
?>

本人眼拙,如有任何错误,可以及时与我联系并及时修正.

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在UVM(Universal Verification Methodology)中,callback回调函数是一种在特定事件发生时自动调用的函数。这些事件可以是UVM中定义的一些特定动作,例如创建或删除一个对象,或者在测试用例的不同阶段执行某些操作。 UVM中使用callback回调函数的目的是在特定事件发生时执行一些自定义的操作,而不需要修改UVM的源代码。这样可以提高代码的可重用性和灵活性。 在UVM中,callback回调函数通常是通过使用UVM提供的`uvm_callback`类来实现的。这个类是一个基类,用户可以从这个基类派生出自己的回调类,并重载其中的虚拟函数来实现自定义的操作。例如,用户可以派生一个自己的回调类,并重载`execute`函数来定义在某个特定事件发生时执行的操作。 以下是一个示例代码片段,展示了如何在UVM中使用callback回调函数: ```systemverilog class my_callback extends uvm_callback; // 重载execute函数来定义回调函数的操作 virtual function void execute(uvm_object obj); // 在这里编写自定义的操作 $display("Callback function called!"); endfunction endclass // 在需要使用callback回调函数的地方创建一个回调对象 my_callback callback; // 注册回调对象到对应的事件上 my_object.add_callback(callback); // 当事件发生时,回调函数会被自动调用 ``` 在上述示例中,我们创建了一个名为`my_callback`的回调类,并重载了`execute`函数。然后,我们创建了一个回调对象`callback`,并将其注册到一个特定的事件上(`my_object.add_callback(callback)`)。当该事件发生时,execute`函数会被自动调用,并执行自定义的操作。 请注意,使用callback回调函数需要遵循UVM的规范和最佳实践。在实际使用中,可以根据具体需求来设计和实现自己的callback回调函数。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值