> 백엔드 개발 > PHP 튜토리얼 > PHP 콜백 함수와 익명 함수의 상세한 해석

PHP 콜백 함수와 익명 함수의 상세한 해석

墨辰丷
풀어 주다: 2023-03-26 16:04:01
원래의
1343명이 탐색했습니다.

이 글은 주로 PHP 콜백 함수와 익명 함수를 소개하고, PHP 콜백 함수와 익명 함수의 구체적인 기능과 사용법, 관련 주의사항을 예시로 분석해 놓았으니 필요한 친구들이 참고할 수 있습니다

자세한 내용은 다음과 같습니다.

콜백 함수 및 익명 함수

JS에서는 콜백 함수와 클로저를 사용하여 이벤트 메커니즘을 완성하고 많은 복잡한 작업을 수행할 수 있습니다. PHP에서는 일반적으로 사용되지 않습니다. 오늘은 PHP의 콜백 함수와 익명 함수에 대해 이야기하겠습니다.

콜백 함수

콜백 함수: 콜백(즉, 콜백은 메인 함수에 의해 호출되고 작동된 후 메인 함수로 돌아갑니다)은 다음과 같은 특정 실행 코드 조각에 대한 참조를 나타냅니다. 함수 매개변수를 통해 다른 코드로 전달됩니다.

사용하기 위해 함수를 다른 함수에 매개변수로 전달하는 것이 널리 사용됩니다. PHP에는 array_map, usort, call_user_func_array 등과 같이 "매개변수를 함수로 요구하는" 함수가 많이 있습니다. 그런 다음 결과를 직접 기본 함수로 반환합니다. 장점은 함수를 값으로 사용하기 편리하고, 코드가 간결하고 가독성이 좋다는 점입니다.

익명 함수

익명 함수는 이름에서 알 수 있듯이 정해진 함수 이름이 없는 함수입니다. PHP는 익명 함수와 클로저를 동일한 개념으로 취급합니다(PHP에서는 익명 함수를 클로저 함수라고도 합니다). 물론 사용법은 변수로만 사용할 수 있습니다.

PHP에서 변수에 함수를 할당하는 방법에는 네 가지가 있습니다.

① 우리가 자주 사용하는 방법: 함수는 외부에서/또는 PHP에 내장되어 정의되며, 함수 이름은 문자열 매개변수로 직접 전달됩니다. 참고: 클래스 정적 함수인 경우 CLASS::FUNC_NAME으로 전달되어야 합니다. CLASS::FUNC_NAME的方式传入。

② 使用create_function($args, $func_code);创建函数,会返回一个函数名。 $func_code为代码体,$args为参数字符串,以','分隔;

③ 直接赋值:$func_name = function($arg){statement}

④ 直接使用匿名函数,在参数处直接定义函数,不赋给具体的变量值;

第一种方式因为是平常所用,不再多提;第二种类似eval()方法的用法,也被PHP官方列为不推荐使用的方式,而且其定义方式太不直观,我除了测试外,也没有在其他地方使用过,也略过不提。在这里重点说一下第三种和第四种用法;

后两种创建的函数就被称为匿名函数,也就是闭包函数, 第三种赋值法方式创建的函数非常灵活,可以通过变量引用。可以用 is_callable($func_name) 来测试此函数是否可以被调用, 也可以通过$func_name($var)来直接调用;而第四种方式创建的函数比较类似于JS中的回调函数,不需要变量赋值,直接使用;

另外要特别介绍的是 use 关键词,它可以在定义函数时,用来引用父作用域中的变量;用法为 function($arg) use($outside_arg) {function_statement} 。其中$outside_arg 为父作用域中的变量,可以在function_statement使用。

这种用法用在回调函数“参数值数量确定”的函数中。 如usort需求$callback的参数值为两项,可是我们需要引入别的参数来影响排序怎么办呢?使用use()关键词就很方便地把一个新的变量引入$callback内部使用了。

array_map/array_filter/array_walk:

把这三个函数放在一块是因为这三个函数在执行逻辑上比较类似,类似于下面的代码:

$result = [];
foreach($vars as $key=>$val){
  $item = callback();
  $result[] = $item;
}
return $result;
array_walk($vars, $callback)
로그인 후 복사

其callback应如下:

$callback = function(&$val, $key[, $arg]){  
  doSomething($val);
}
로그인 후 복사

array_walk返回执行是否成功,是一个布尔值。对$value添加引用符号可以在函数内改变$value值,以达到改变$vars数组的效果。由于其$callback对参数数量要求为两项,array_walk不能传入strtolower/array_filter之类的$callback,若想实现类似功能,可以使用接下来要说的array_map()

create_function($args, $func_code);를 사용하여 함수 이름을 반환하는 함수를 생성하세요. $func_code는 코드 본문이고, $args는 ','로 구분된 매개변수 문자열입니다.

3 직접 할당: $func_name = function($arg){statement}

4 익명 함수 사용 직접, 특정 변수 값을 할당하지 않고 매개변수에서 직접 함수를 정의합니다.

첫 번째 방법은 일반적으로 사용되므로 더 이상 언급되지 않습니다. 두 번째 방법은 eval() 방법과 유사하며 공식적으로는 부적절하다고 나열됩니다. by PHP 추천하는 방법이고, 정의가 너무 직관적이지 않습니다. 테스트 외에는 다른 곳에서는 사용해본 적이 없어서 언급하지 않겠습니다. 여기서는 세 번째와 네 번째 사용법에 중점을 둡니다.

후자의 두 가지에 의해 생성된 함수는 익명 함수, 즉 세 번째 할당 방법으로 생성된 함수는 매우 유연하며 변수를 통해 전달될 수 있습니다. is_callable($func_name)을 사용하여 이 함수를 호출할 수 있는지 테스트하거나 $func_name($var)을 통해 직접 호출할 수 있습니다. 네 번째 방법으로 생성된 함수는 JS의 콜백 함수와 더 유사하며 그렇지 않습니다.

또 다른 특별한 소개는 함수를 정의할 때 상위 범위의 변수를 참조하는 데 사용할 수 있는 키워드입니다. function($arg) use($outside_arg) {function_statement} . 그 중 $outside_arg는 상위 스코프의 변수로 function_statement에서 사용할 수 있습니다.

이 사용법은 "매개변수 값의 개수가 결정됩니다"라는 콜백 함수에서 사용됩니다. 예를 들어 usort에서는 $callback의 매개변수 값이 두 개의 항목이어야 하지만 정렬에 영향을 주기 위해 다른 매개변수를 도입해야 하는 경우에는 어떻게 될까요? use() 키워드를 사용하면 내부 사용을 위해 $callback에 새 변수를 도입하는 것이 매우 편리합니다.

array_map/array_filter/array_walk:

이 세 함수의 실행 논리가 상대적으로 유사하기 때문에 이 세 함수를 함께 사용하세요. 다음 코드와 비슷합니다.

$callback = function($var){
  return true or false;     
}
로그인 후 복사

its 콜백은 다음과 같아야 합니다. 다음:

$callback = function($var_a[, $var_b...]){
  doSomething($var_a, $var_b);
}
로그인 후 복사

array_walk는 실행 성공 여부를 부울 값으로 반환합니다. $value에 참조 기호를 추가하면 함수 내의 $value 값을 변경하여 $vars 배열을 변경하는 효과를 얻을 수 있습니다. $callback에는 두 개의 매개변수가 필요하므로 array_walk는 strtolower/array_filter와 같은 $callback을 전달할 수 없습니다. 유사한 기능을 수행하려면 다음에 언급된 array_map()을 사용할 수 있습니다. 🎜🎜🎜array_walk_recursive($arr, $callback);🎜🎜🎜반환 값과 실행 메커니즘은 array_walk와 유사합니다.🎜🎜콜백은 array_walk와 동일하지만 차이점은 $val이 배열인 경우 함수는 $val을 아래쪽으로 재귀적으로 처리합니다. 이 경우 $val은 배열의 $key이며 무시됩니다. 🎜🎜🎜array_filter($vars, $callback, $flag);🎜🎜🎜its $callback은 다음과 유사합니다. 🎜🎜🎜
callback = function($left, $right){
    $res = compare($left, $right);
    return $res;
}
로그인 후 복사
로그인 후 복사
🎜🎜🎜array_filter는 $callback이 실행될 때 false를 반환하는 항목을 필터링하고 array_filter는 필터링이 완료된 배열입니다. 🎜🎜세 번째 매개변수인 $flag는 콜백 매개변수인 $var의 값을 결정하지만, 이는 PHP 상위 버전의 기능일 수 있으며 직접 테스트할 수 있습니다. 기본적으로 배열의 각 항목 값이 전달됩니다. 플래그가 ARRAY_FILTER_USE_KEY이면 배열의 각 항목의 키가 전달되고 ARRAY_FILTER_USE_BOTH는 키와 값에 전달됩니다. 🎜🎜🎜array_map($callback; , &$var_as [,$var_bs...]);🎜 🎜🎜its $callback은 다음과 유사합니다: 🎜🎜🎜
$callback = function($initial, $var){
    $initial = calculate($initail, $var);
    return $initial;
}
로그인 후 복사
로그인 후 복사
🎜🎜

返回$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;
});
로그인 후 복사

相关推荐:

PHP回调函数与匿名函数使用案例解析

结合代码详细为你讲解,php中的array_map,array_walk以及匿名函数

php中的匿名函数和闭包(closure)用法

위 내용은 PHP 콜백 함수와 익명 함수의 상세한 해석의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

관련 라벨:
원천:php.cn
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
인기 튜토리얼
더>
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿