Что-то вроде функции делегата обратного вызова в php - PullRequest
15 голосов
/ 07 сентября 2008

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

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

Ответы [ 3 ]

16 голосов
/ 07 сентября 2008

(кроме схемы наблюдения) вы также можете использовать 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') );
?>
4 голосов
/ 07 сентября 2008

Как вы относитесь к использованию шаблона 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's name. 
doIt('myCallback'); 

Показывает: Данные есть: это мои данные

3 голосов
/ 01 июля 2010

Мне было интересно, можем ли мы использовать магический метод __invoke, чтобы создать «первоклассную» функцию первого класса и таким образом реализовать обратный вызов

Звук примерно такой, для 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 для такой функции.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...