枕草子
いつも畏敬の念を抱いてください。
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_recursive($arr, $callback);
戻り値と実行メカニズムはarray_walkに似ています;
コールバックはarray_walkと同じですが、違いは、$valが配列の場合、関数は$val を下方向に再帰的に処理します。この場合、$val は配列の $key であり、無視されることに注意してください。
array_filter($vars, $callback, $flag);
その $callback は次のようになります:
$callback = function($var){ return true or false; }
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,介绍了几个用法~希望对大家有帮助,如果有问题,欢迎指出,如果您喜欢,可以点下推荐~