Прийняти функцію як параметр у PHP


104

Мені було цікаво, чи можна чи ні передавати функцію як параметр у PHP; Я хочу щось подібне, коли ви програмуєте в JS:

object.exampleMethod(function(){
    // some stuff to execute
});

Що я хочу - це виконувати цю функцію десь у exampleMethod. Це можливо в PHP?


Відповіді:


150

Це можливо, якщо ви використовуєте PHP 5.3.0 або вище.

Див. Анонімні функції в посібнику.

У вашому випадку ви б визначили exampleMethodтак:

function exampleMethod($anonFunc) {
    //execute anonymous function
    $anonFunc();
}

9
Чорт, я знав, що це можливо. Збирався відповісти, але хотів знайти якусь документацію, до якої посилатись спочатку, і не знав, як саме називається. Ну добре, тепер я знаю, коли мені також потрібно це зробити. Дякую.
Роб

3
До 5.3 ви можете скористатисяcreate_function
Гордон

Дуже дякую. Оскільки я повинен це зробити для PHP4.3, я думаю, мені доведеться робити те, що я хочу, використовуючи іншу логіку.
Крістіан

1
@Casidiablo - Дивіться відповідь Джейга, можливо, вам вдасться щось зробити create_function(). Це не зовсім те саме, що вам потрібно передати свою функцію як рядок коду, який потім потрапляє eval()за лаштунки. Не ідеально, але ви, можливо, зможете ним скористатися.
зомбат

1
Як би ви це назвали? Чи можете ви назвати існуючу функцію? Напр .:exampleMethod(strpos);
сумід

51

Просто для додавання до інших, ви можете передати ім'я функції:

function someFunc($a)
{
    echo $a;
}

function callFunc($name)
{
    $name('funky!');
}

callFunc('someFunc');

Це буде працювати в PHP4.


17

Ви також можете використовувати create_function для створення функції як змінної та передачі її навколо. Хоча мені подобається відчуття анонімних функцій краще. Іди зомбатом.


+1 для альтернативи. Здається, що ОП може знадобитися.
зомбат

Дякую! Я повинен дотримуватися старої установки PHP 5.2, а функції анонімних функцій там не працюють.
діосней

Попередження: Цю функцію було знято з PHP 7.2.0. Покладатися на цю функцію дуже не рекомендується.
shamaseen

15

Просто кодуйте це так:

function example($anon) {
  $anon();
}

example(function(){
  // some codes here
});

було б чудово, якби ви могли придумати щось подібне (надихнувши Laravel Illuminate):

Object::method("param_1", function($param){
  $param->something();
});

саме те, що я шукав
Харві

5

PHP VERSION> = 5.3.0

Приклад 1: основний

function test($test_param, $my_function) {
    return $my_function($test_param);
}

test("param", function($param) {
    echo $param;
}); //will echo "param"

Приклад 2: об’єкт std

$obj = new stdClass();
$obj->test = function ($test_param, $my_function) {
    return $my_function($test_param);
};

$test = $obj->test;
$test("param", function($param) {
    echo $param;
});

Приклад 3: нестатичний виклик класу

class obj{
    public function test($test_param, $my_function) {
        return $my_function($test_param);
    }
}

$obj = new obj();
$obj->test("param", function($param) {
    echo $param;
});

Приклад 4: виклик статичного класу

class obj {
    public static function test($test_param, $my_function) {
        return $my_function($test_param);
    }
}

obj::test("param", function($param) {
    echo $param;
});

5

Відповідно до відповіді @ zombat, краще спочатку перевірити анонімні функції:

function exampleMethod($anonFunc) {
    //execute anonymous function
    if (is_callable($anonFunc)) {
        $anonFunc();
    }
}

Або підтвердити тип аргументу з PHP 5.4.0:

function exampleMethod(callable $anonFunc) {}

як вам впоратися, коли перевірка (callable $ anonFunc) не вдалася? спасибі
xam

3

Перевірено на PHP 5.3

Як я бачу тут, Anonymous Function може допомогти вам: http://php.net/manual/en/functions.anonymous.php

Те, що вам, мабуть, знадобиться, і раніше не буде сказано, як передавати функцію, не загортаючи її всередині створеної на льоту функції . Як ви побачите пізніше, вам потрібно буде передати ім'я функції, записане в рядку як параметр, перевірити його "calllability" і потім викликати його.

Функція перевірки:

if( is_callable( $string_function_name ) ){
    /*perform the call*/
}

Потім, щоб викликати його, використовуйте цей фрагмент коду (якщо вам також потрібні параметри, покладіть їх на масив), який можна побачити за адресою: http://php.net/manual/en/function.call-user-func.php

call_user_func_array( "string_holding_the_name_of_your_function", $arrayOfParameters );

як це випливає (аналогічним, параметричним способом):

    function funToBeCalled(){
        print("----------------------i'm here");
    }
    function wrapCaller($fun){
        if( is_callable($fun)){
            print("called");
            call_user_func($fun);
        }else{
            print($fun." not called");
        }
    }

    wrapCaller("funToBeCalled");
    wrapCaller("cannot call me");

Ось клас, що пояснює, як зробити щось подібне:

<?php
class HolderValuesOrFunctionsAsString{
    private $functions = array();
    private $vars = array();

    function __set($name,$data){
        if(is_callable($data))
            $this->functions[$name] = $data;
        else
            $this->vars[$name] = $data;
    }

    function __get($name){
        $t = $this->vars[$name];
        if(isset($t))
            return $t;
        else{
            $t = $this->$functions[$name];
            if( isset($t))
                return $t;
        }
    }

    function __call($method,$args=null){
        $fun = $this->functions[$method];
        if(isset($fun)){
            call_user_func_array($fun,$args);
        } else {
            // error out
            print("ERROR: Funciton not found: ". $method);
        }
    }
}
?>

і приклад використання

<?php
    /*create a sample function*/
    function sayHello($some = "all"){
    ?>
         <br>hello to <?=$some?><br>
    <?php
    }

    $obj = new HolderValuesOrFunctionsAsString;

    /*do the assignement*/
    $obj->justPrintSomething = 'sayHello'; /*note that the given
        "sayHello" it's a string ! */

    /*now call it*/
    $obj->justPrintSomething(); /*will print: "hello to all" and
        a break-line, for html purpose*/

    /*if the string assigned is not denoting a defined method
         , it's treat as a simple value*/
    $obj->justPrintSomething = 'thisFunctionJustNotExistsLOL';

    echo $obj->justPrintSomething; /*what do you expect to print?
        just that string*/
    /*N.B.: "justPrintSomething" is treated as a variable now!
        as the __set 's override specify"*/

    /*after the assignement, the what is the function's destiny assigned before ? It still works, because it's held on a different array*/
     $obj->justPrintSomething("Jack Sparrow");


     /*You can use that "variable", ie "justPrintSomething", in both ways !! so you can call "justPrintSomething" passing itself as a parameter*/

     $obj->justPrintSomething( $obj->justPrintSomething );
         /*prints: "hello to thisFunctionJustNotExistsLOL" and a break-line*/

    /*in fact, "justPrintSomething" it's a name used to identify both
         a value (into the dictionary of values) or a function-name
         (into the dictionary of functions)*/
?>

2

Простий приклад використання класу:

class test {

    public function works($other_parameter, $function_as_parameter)
    {

        return $function_as_parameter($other_parameter) ;

    }

}

$obj = new test() ;

echo $obj->works('working well',function($other_parameter){


    return $other_parameter;


});
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.