Что-то вроде функции делегата callback в php

Я хотел бы реализовать что-то похожее на метод делегата С# в PHP. Быстрое объяснение того, что я пытаюсь сделать в целом: я пытаюсь реализовать некоторые асинхронные функции. В основном некоторые ресурсоемкие вызовы, которые попадают в очередь, кэшируются и отправляются, когда базовая система приближается к ней. Когда асинхронный вызов, наконец, получит ответ, я бы хотел, чтобы событие обратного вызова было поднято.

У меня возникли проблемы с механизмом выполнения обратных вызовов в PHP. Я придумал метод, который работает сейчас, но я недоволен этим. В основном это включает передачу ссылки на объект и имя метода на нем, который будет служить в качестве обратного вызова (принимая ответ в качестве аргумента), а затем использовать eval для вызова метода, когда это необходимо. Это неоптимально по целому ряду причин, есть ли лучший способ сделать это, о котором все знают?

Ответы

Ответ 1

(Помимо шаблона наблюдателя) вы также можете использовать call_user_func() или call_user_func_array().

Если вы передадите array(obj, methodname) в качестве первого параметра, он будет вызываться как $obj->methodname().

<?php
class Foo {
    public function bar($x) {
        echo $x;
    }
}

function xyz($cb) {
    $value = rand(1,100);
    call_user_func($cb, $value);
}

$foo = new Foo;
xyz( array($foo, 'bar') );
?>

Ответ 2

Как вы относитесь к использованию шаблона Observer? Если нет, вы можете реализовать настоящий обратный вызов следующим образом:

// 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 .  "\n"; 
} 


// Call doIt() and pass our sample callback function name. 
doIt('myCallback'); 

Дисплеи: данные: это мои данные

Ответ 3

Мне было интересно, можем ли мы использовать метод __invoke magic для создания "рода" функции первого класса и, таким образом, реализовать обратный вызов

Звучит примерно так, для PHP 5.3

interface Callback 
{
    public function __invoke(); 
}

class MyCallback implements Callback
{
    private function sayHello () { echo "Hello"; }
    public function __invoke ()  { $this->sayHello(); } 
}

class MySecondCallback implements Callback
{
    private function sayThere () { echo "World"; }
    public function __invoke ()  { $this->sayThere(); }
}

class WhatToPrint
{
    protected $callbacks = array();
    public function register (Callback $callback) 
    {
        $this->callbacks[] = $callback;
        return $this;
    }
    public function saySomething ()
    {
        foreach ($this->callbacks as $callback) $callback(); 
    }
}

$first_callback = new MyCallback;
$second_callback = new MySecondCallback;
$wrapper = new WhatToPrint;
$wrapper->register($first_callback)->register($second_callback)->saySomething();

Будет напечатан HelloWorld

Надеюсь, это поможет;)

Но я бы предпочел шаблон Controller с SPL для такой функции.