Home > Backend Development > PHP Tutorial > PHP core features anonymous functions

PHP core features anonymous functions

藏色散人
Release: 2023-04-07 18:22:01
forward
2806 people have browsed it

Proposed

Before the emergence of anonymous functions, all functions needed to be named before they could be used

function increment($value)
{
    return $value + 1;
}
array_map('increment', [1, 2, 3]);
Copy after login

Sometimes a function may only need to be used once. Using anonymous functions will make the code more concise and intuitive, and also prevent the function from being used elsewhere

array_map(function($value){
    return $value + 1;
}, [1, 2, 3]);
Copy after login

Definition and use

PHP treats closures and anonymous functions as are equivalent concepts (collectively referred to as anonymous functions in this article), and are essentially objects disguised as functions.

The essence of anonymous functions is objects, so just like objects, anonymous functions can be assigned to a variable

$greet = function(string $name){
    echo "hello {$name}";
}
$greet("jack") // hello jack
Copy after login

All anonymous functions are instances of Closure objects

$greet instanceof Closure // true
Copy after login

The object has no parent scope to speak of, so you need to use use to manually declare the variables used.

$num = 1;
$func = function() use($num){
    $num = $num + 1;
    echo $num;
}
$func();  // 2
echo $num;  // 还是 1
Copy after login

If you want the variables in the anonymous function to take effect, you need to use reference pass-by-value

$num = 1;
$func = function() use(&$num){
    $num = $num + 1;
    echo $num;
}
$func();  // 2
echo $num;  // 2
Copy after login

Starting from PHP 5.4, when using an anonymous function in a class, $this of the anonymous function will be automatically bound to the current class

class Foo {
    public function bar()
    {   
        return function() {
            return $this;
        };
    }
}
$foo = new Foo();
$obj = $foo->bar(); // Closure()
$obj();   // Foo
Copy after login

If you do not want the automatic binding to take effect, you can use a static anonymous function

class Foo {
    public function bar()
    {   
        return static function() {
            return $this;
        };
    }
}
$foo = new Foo();
$obj = $foo->bar(); // Closure()
$obj();   // Using $this when not in object context
Copy after login

The essence of anonymous functions

The essence of anonymous functions is the Closure object, including the following five methods

Closure {
    private __construct ( void )
    public static bind ( Closure $closure , object $newthis [, mixed $newscope = "static" ] ) : Closure
    public bindTo ( object $newthis [, mixed $newscope = "static" ] ) : Closure
    public call ( object $newthis [, mixed $... ] ) : mixed
    public static fromCallable ( callable $callable ) : Closure
}
Copy after login

__construct - Prevent anonymous functions from being instantiated

$closure = new \Closure();
// PHP Error:  Instantiation of 'Closure' is not allowed
Copy after login

Closure::bindTo - Copies the current anonymous function object and binds the specified $this object and class scope. In layman's terms, it means manually binding an anonymous function to a specified object. Using this, you can extend the functionality of the object.

// 定义商品类
class Good {
    private $price;
    public function __construct(float $price)
    {
        $this->price = $price;
    }
}
// 定义一个匿名函数,计算商品的促销价
$addDiscount = function(float $discount = 0.8){
    return $this->price * $discount;
}
$good = new Good(100);
// 将匿名函数绑定到 $good 实例,同时指定作用域为 Good
$count = $addDiscount->bindTo($good, Good::class); 
$count(); // 80
// 将匿名函数绑定到 $good 实例,但是不指定作用域,将无法访问 $good 的私有属性
$count = $addDiscount->bindTo($good); 
$count(); // 报错
Copy after login

Closure::bind - A static version of the bindTo method, with two usages:

Usage 1: To achieve the same effect as the bindTo method

$count = \Closure::bind($addDiscount, $good, Good::class);
Copy after login

Usage 2: Bind the anonymous function to the class (rather than the object), remember to set the second parameter to null

// 商品库存为 10
class Good {
    static $num = 10;
}
// 每次销售后返回当前库存
$sell = static function() {
    return"当前库存为". --static::$num ;
};
// 将静态匿名函数绑定到 Good 类中
$sold = \Closure::bind($sell, null, Good::class);
$sold(); // 当前库存为 9
$sold(); // 当前库存为 8
Copy after login

call - PHP 7 New The call method can bind and call anonymous functions. In addition to simpler syntax, the performance is also higher

// call 版本
$addDiscount->call($good, 0.5);  // 绑定并传入参数 0.5,结果为 50
// bindTo 版本
$count = $addDiscount->bindTo($good, Good::class, 0.5); 
$count(); // 50
Copy after login

fromCallable - Convert a given callable function into an anonymous function

class Good {
    private $price;
    public function __construct(float $price)
    {
        $this->price = $price;
    }
}
function addDiscount(float $discount = 0.8){
    return $this->price * $discount;
}
$closure = \Closure::fromCallable('addDiscount');
$good = new Good(100);
$count = $closure->bindTo($good);  
$count = $closure->bindTo($good, Good::class);   // 报错,不能重复绑定作用域
$count(); // 报错,无法访问私有属性
Copy after login

fromCallable is equivalent to

$reflexion = new ReflectionFunction('addDiscount');
$closure = $reflexion->getClosure();
Copy after login

One thing that needs special attention here is that whether it is a closure converted fromCallable or a closure obtained using reflection, when using bindTo, if the second If the parameter specifies the binding class, an error will be reported

Cannot rebind scope of closure created by ReflectionFunctionAbstract::getClosure()
Copy after login

The above is the detailed content of PHP core features anonymous functions. For more information, please follow other related articles on the PHP Chinese website!

Related labels:
php
source:learnku.com
Statement of this Website
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn
Popular Tutorials
More>
Latest Downloads
More>
Web Effects
Website Source Code
Website Materials
Front End Template