ホームページ  >  記事  >  バックエンド開発  >  PHP のコールバック関数と匿名関数

PHP のコールバック関数と匿名関数

高洛峰
高洛峰オリジナル
2016-11-15 15:25:261021ブラウズ

枕草子

いつも畏敬の念を抱いてください。

PHPのコールバック関数と匿名関数

はじめに

少し前まで会社が忙しくて、毎日仕事から帰ってくると頭が重く感じていました。土日は色々あって、コーディングを始めたくないのでブログを後回しにしてましたが、近いうちに忙しくなりそうな予感がしたので始めます。私の収穫を書き、要約し、共有する時間がありますので、注目してください。

コールバック関数と匿名関数

コールバック関数とクロージャは JS にとって馴染みのないものではなく、JS はそれらを使用してイベント メカニズムを完成させ、多くの複雑な操作を実行できます。 PHP では一般的には使用されません。今日は、PHP のコールバック関数と匿名関数について説明します。

コールバック関数

コールバック関数: コールバック (つまり、コールバックは、メイン関数によって呼び出され操作された後、メイン関数に戻ります) は、他のコードに渡される実行可能コードの特定の部分への参照を指します。関数パラメータを通じて。

一般的な説明は、関数をパラメータとして別の関数に渡して使用するというものです。PHP には、array_map、usort、call_user_func_array など、「関数としてパラメータを必要とする」関数が多数あります。これらは、渡された関数を実行し、その後、結果を main 関数に直接返します。利点は、関数を値として使用するのが便利で、コードが簡潔で読みやすいことです。

匿名関数:

匿名関数は、名前が示すように、決まった関数名を持たない関数です。PHP では、匿名関数とクロージャを同じ概念として扱います (匿名関数は、PHP ではクロージャ関数とも呼ばれます)。もちろん、その使用法は変数としてのみ使用できます。

PHP で変数に関数を割り当てるには 4 つの方法があります:

私たちがよく使う方法: 関数は外部/または PHP の組み込みで定義され、関数名は文字列パラメーターとして直接渡されます。注: クラス静的関数の場合は、CLASS::FUNC_NAME として渡されます。

create_function($args, $func_code); を使用して関数を作成し、関数名を返します。 $func_code はコード本体、$args は「,」で区切られたパラメータ文字列です。

直接代入: $func_name = function($arg){statement};

匿名関数を直接使用し、関数を直接定義します。パラメータには特定の変数値を割り当てないでください。最初のメソッドは一般的に使用されているため、これ以上言及しません。2 番目のメソッドも、eval() メソッドと同様に、PHP によって非推奨のメソッドとして正式にリストされています。定義 この方法はあまりにも直感的ではないので、テスト以外の場所では使用したことがないため、言及しません。ここでは、3 番目と 4 番目の使用法に焦点を当てます。

後の 2 つによって作成された関数は、匿名関数、つまりクロージャ関数と呼ばれます。3 番目の代入メソッドによって作成された関数は、非常に柔軟であり、変数を介して渡すことができます。 is_callable($func_name) を使用してこの関数を呼び出せるかどうかをテストすることも、$func_name($var) を通じて直接呼び出すこともできます。4 番目の方法で作成された関数は、JS のコールバック関数に似ていますが、変数の割り当てが必要です。

もう 1 つの特別な導入は、関数を定義するときに親スコープ内の変数を参照するために使用できる、 function($arg) use($outside_arg) {function_statement} です。 。このうち、$outside_arg は親スコープ内の変数であり、function_statement 内で使用できます。

この使い方は「パラメータ値の数を決定する」コールバック関数で使用されます。 たとえば、usort では $callback のパラメータ値が 2 つの項目である必要がありますが、並べ替えに影響を与えるために他のパラメータを導入する必要がある場合はどうすればよいでしょうか? use() キーワードを使用すると、内部使用のために $callback に新しい変数を簡単に導入できます。

array_map/array_filter/array_walk:

これら 3 つの関数の実行ロジックは比較的似ており、次のコードに似ているため、これら 3 つの関数をまとめます:

$result = [];
foreach($vars as $key=>$val){
    $item = callback();
    $result[] = $item;
}
return $result;

array_walk($vars, $callback)

コールバックは次のようになります。次のように:

$callback = function(&$val, $key[, $arg]){    
            doSomething($val);
        }

array_walk は、実行が成功したかどうかをブール値で返します。 $value に参照シンボルを追加すると、関数内の $value 値を変更して、$vars 配列を変更する効果を得ることができます。 $callback には 2 つのパラメータが必要なため、array_walk は strto lower/array_filter などの $callback を渡すことができません。同様の関数を実現したい場合は、次に説明する array_map() を使用できます。

array_walk_recursive($arr, $callback);


戻り値と実行メカニズムはarray_walkに似ています;


コールバックはarray_walkと同じですが、違いは、$valが配列の場合、関数は$val を下方向に再帰的に処理します。この場合、$val は配列の $key であり、無視されることに注意してください。


array_filter($vars, $callback, $flag);


その $callback は次のようになります:

$callback = function($var){
              return true or false;         
            }

array_filter は、$callback の実行時に false を返す項目を除外し、array_filter はフィルタリング後の配列を返します。完成されました。


3 番目のパラメータ $flag はコールバック パラメータ $var の値を決定しますが、これは PHP の上位バージョンの機能である可能性があります。私の PHP5.5.3 はそれをサポートしていません。デフォルトでは、配列内の各項目の値が渡されます。フラグが ARRAY_FILTER_USE_KEY の場合、配列内の各項目のキーが渡され、ARRAY_FILTER_USE_BOTH がキーと値に渡されます。


array_map( $callback, &$var_as [,$var_bs...]) ;


その $callback は次のようになります:

$callback = function($var_a[, $var_b...]){
            doSomething($var_a, $var_b);
        }

返回$var_as经过callback处理后的数组(会改变原数组);如果有多个数组的时候将两个数组同样顺序的项目传入处理,执行次数为参数数组中项目最多的个数;

usort/array_reduce

把这两个函数放在一块,因为他们的执行机制都有些特殊。

usort(&$vars, $callback)

$callback应该如下:

    callback = function($left, $right){
        $res = compare($left, $right);
        return $res;
    }

usort返回执行成功与否,bool值。用户自定义方法 比较$left 和 $right,其中$left和$right是$vars中的任意两项;

$left > $right时返回 正整数, $left < $right时返回 负整数, $left = $right时返回0;

$vars中的元素会被取出会被由小到大升序排序。 想实现降序排列,将$callback的返回值反一下就行了。

array_reduce($vars ,$callable [, mixed $initial = NULL])

$callback应该如下:

    $callback = function($initial, $var){
        $initial = calculate($initail, $var);
        return $initial;
    }

初始值$initial默认为null,返回经过迭代后的initial;一定要将$initial返回,这样才能不停地改变$initial的值,实现迭代的效果。

这里顺便说一下map和reduce的不同:

map:将数组中的成员遍历处理,每次返回处理后的一个值,最后结果值为所有处理后值组成的多项数组;

reduce:遍历数组成员,每次使用数组成员结合初始值处理,并将初始值返回,即使用上一次执行的结果,配合下一次的输入继续产生结果,结果值为一项;

call_user_func/call_user_func_array

call_user_func[_array]($callback, $param)


$callback形如:

    $callback = function($param){
        $result = statement(); 
        return $result;
    }

返回值多种,具体看$callback。

可用此函数实现PHP的事件机制,其实并不高深,在判断条件达成,或程序执行到某一步后 call_user_func()就OK了。这个我在之前的博客中也有介绍到:搭建自己的PHP框架心得(二)

总结

其实以上$callback不用单独定义并使用变量引用,使用上面说过的第四种函数定义方式,直接在函数内定义,使用‘完全’匿名函数就行了。 如:

usort($records, function mySortFunc($arg) use ($order){
    func_statement;
});

是不是逼格满满呢?

OK,介绍了几个用法~希望对大家有帮助,如果有问题,欢迎指出,如果您喜欢,可以点下推荐~


声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。