>  기사  >  백엔드 개발  >  PHP의 익명 함수와 클로저란 무엇입니까? 만드는 방법?

PHP의 익명 함수와 클로저란 무엇입니까? 만드는 방법?

PHPz
PHPz앞으로
2016-07-29 08:33:421963검색


이 글의 내용은 PHP의 익명 함수와 클로저가 무엇인지 소개하는 것입니다. 익명 함수와 클로저를 만드는 방법은 무엇입니까? 도움이 필요한 친구들이 참고할 수 있기를 바랍니다.

개요

클로저와 익명 함수는 PHP 5.3.0에서 도입되었습니다. 이 두 기능은 모든 PHP 개발자에게 매우 유용합니다. 마스터했습니다.

익명 함수는 실제로 이름이 없는 함수입니다. 익명 함수는 변수에 할당될 수 있으며 다른 PHP 함수 개체처럼 전달될 수 있습니다. 그러나 익명 함수는 여전히 함수이므로 호출할 수 있고 매개변수를 전달할 수 있으므로 함수나 메서드에 대한 콜백으로 적합합니다.

클로저란 클로저가 생성될 때 주변 상태를 캡슐화하는 기능을 의미합니다. 클로저가 위치한 환경이 더 이상 존재하지 않더라도 클로저에 캡슐화된 상태는 여전히 존재합니다.

익명 함수 생성

익명 함수 생성은 간단합니다.

//将匿名函数赋给一个变量,通过变量名+()的形式来调用
$greet = function () {
    return "Hello World";
};

echo $greet();

결과 인쇄:

Hello World

익명 함수는 일반 PHP 함수와 매우 유사합니다. 공통 구문이 동일하고 매개변수도 허용하며 값을 반환할 수 있습니다. 그러나 클로저에는 함수 이름이 없습니다.

참고: $greet 변수를 호출할 수 있는 이유는 이 변수의 값이 클로저이고, 뒤에 ()가 있는 한 클로저 개체가 __invoke() 매직 메서드를 구현하기 때문입니다. 변수 이름, PHP __invoke 메소드를 찾아서 호출합니다.

우리는 일반적으로 함수나 메소드의 콜백으로 익명 함수를 사용합니다. 실제로 많은 PHP 함수는 array_mappreg_replace_callback과 같은 익명 함수를 사용합니다. 다른 값과 마찬가지로 클로저도 다른 PHP 함수에 매개변수로 전달될 수 있습니다.

$numberPlusOne = array_map(function ($number) {
    return $number += 1;
}, [1, 2, 3]);

print_r($numberPlusOne);

익명 함수가 등장하기 전에는 PHP 개발자가 이러한 기능을 구현하기 위해 이름이 지정된 함수만 별도로 만든 다음 사용할 수 있었습니다. 함수를 참조하는 이름:

function incrementNumber ($number) {
    return $number += 1;
}

$numberPlusOne = array_map(‘incrementNumber’,  [1, 2, 3]);
print_r($numberPlusOne);

이는 콜백 구현을 사용 위치에서 분리하고 클로저를 사용하여 코드를 보다 간결하게 구현합니다.

클로저 만들기

자유 변수를 포함하는 함수와 이러한 모든 자유 변수에 대한 변수 바인딩을 제공하는 환경을 클로저라고 합니다. .
function makeHelloWorld($name) { 
    $i = 0;
    return function()use($name, &$i){
        echo $name.$i. ' <br>';
        $i++;
    };

}
$hello1 = makeHelloWorld("itbsl");
$hello2 = makeHelloWorld("kevin");
$hello1();
$hello1();
$hello1();
$hello2();

결과 인쇄:

itbsl0 
itbsl1 
itbsl2 
kevin0

상위 범위에서 변수 상속

PHP에서 클로저 객체를 수동으로 호출해야 합니다bindTo 메서드를 사용하거나 use 키워드를 사용하여 상위 범위의 변수와 상태를 PHP 클로저에 연결합니다. 실제 애플리케이션에서는 대부분 use 키워드를 사용하여 구현됩니다.

키워드 사용

실제로 클로저는 Laravel 프레임워크에서도 널리 사용되며 가장 일반적인 것은 경로 정의입니다.

Route::group(['domain' => '{account}.myapp.com'], function () {
    Route::get('user/{id}', function ($account, $id) {
        //
    });
});

여기서 두 함수는 익명 함수입니다. 상위 범위에서 변수를 상속하는 사용 시나리오는 Model.php의 IlluminateDatabaseEloquent 메서드(saveOrFail)와 같은 Laravel의 기본 소스 코드에서도 일반적입니다.

PHP의 익명 함수와 클로저란 무엇입니까? 만드는 방법?

메소드 함수는 트랜잭션을 사용하여 모델 데이터를 데이터베이스에 저장하는 것입니다. 여기서는 익명 함수를 사용하여 저장된 상태를 반환하고 use 키워드를 사용하여 상위 범위의 $options를 클로저에 전달합니다. 이 데이터에 액세스하세요.

또한 Model 클래스의 forceFill 메서드와 같이 여러 상위 범위 변수를 익명 함수에 전달하는 것도 지원합니다.

PHP의 익명 함수와 클로저란 무엇입니까? 만드는 방법?

여러 변수는 쉼표로 구분할 수 있습니다.

bindTo 메소드

앞서 클로저가 객체라고 언급했으므로 클로저 Word에서 $this 키를 사용하여 다음을 수행할 수 있습니다. 클로저의 내부 상태를 가져옵니다. 클로저 객체의 기본 상태는 쓸모가 없습니다. 주목해야 할 것은 __invoke 매직 메소드와 bindTo 메소드입니다.

__invoke의 역할에 대해서는 앞서 언급한 바 있습니다. 함수를 호출하여 객체를 호출하려고 하면 자동으로 __invoke() 메서드가 호출됩니다.

다음으로 클로저의 내부 상태를 다른 객체에 바인딩할 수 있는 bindTo 메서드를 살펴보겠습니다. 여기서 bindTo 메소드의 두 번째 매개변수는 바인딩된 클로저의 객체가 속한 PHP 클래스를 지정하는 것입니다. 이러한 방식으로 클로저를 보호하고 객체의 다른 위치에서 액세스할 수 있습니다. 바운드 클로저.

PHP 프레임워크는 bindTo 메서드를 사용하여 라우팅 URL을 익명 콜백 함수에 매핑하는 경우가 많습니다. 따라서 프레임워크는 익명 콜백 함수를 애플리케이션 개체에 바인딩하므로 $this을 사용할 수 있습니다. 🎜>키워드는 중요한 애플리케이션 개체를 나타냅니다.

class App {
    protected $routes = [];
    protected $responseStatus = '200 OK';
    protected $responseContentType = 'text/html';
    protected $responseBody = 'Laravel学院';

    public function addRoute($routePath, $routeCallback) {
        $this->routes[$routePath] = $routeCallback->bindTo($this, __CLASS__);
    }

    public function dispatch($currentPath) {
        foreach ($this->routes as $routePath => $callback) {
            if( $routePath === $currentPath) {
                $callback();
            }
        }
        header('HTTP/1.1 ' . $this->responseStatus);
        header('Content-Type: ' . $this->responseContentType);
        header('Content-Length: ' . mb_strlen($this->responseBody));
        echo $this->responseBody;
    }

}

这里我们需要重点关注addRoute方法,这个方法的参数分别是一个路由路径和一个路由回调,dispatch方法的参数是当前HTTP请求的路径,它会调用匹配的路由回调。第9行是重点所在,我们将路由回调绑定到了当前的App实例上。这么做能够在回调函数中处理App实例的状态:

$app = new App();
$app->addRoute(‘user/nonfu’, function(){
    $this->responseContentType = ‘application/json;charset=utf8’;
    $this->responseBody = ‘{“name”:”LaravelAcademy"}';
});
$app->dispatch(‘user/nonfu');

在Larval底层也有用到bindTo方法,详见Illuminate\Support\Traits\Macroable__call方法:闭包和匿名函数(https://laravelacademy.org/post/4341.html)

总结:以上就是本篇文章的全部内容,希望能对大家的学习有所帮助。


성명:
이 기사는 cnblogs.com에서 복제됩니다. 침해가 있는 경우 admin@php.cn으로 문의하시기 바랍니다. 삭제