PHP 7.4 新语法:箭头函数

PHP
短闭包,也叫做箭头函数,是一种用 php 编写的短函数.当向函数中传递闭包时,这个功能是非常有用的,比如使用 array_map 或是 array_filter函数时.

译者注:PHP7.4 计划于今年底发布,请见 Wiki:PHP 基础信息:发行计划

这就是它们看起来的样子:

// Post 对象的集合
$posts = [/* … */];

$ids = array_map(fn($post) => $post->id, $posts);

而以前,你必须这样写:

$ids = array_map(function ($post) {
    return $post->id;
}, $posts);

我们来总结一下短闭包函数如何使用.

  • 在 PHP 7.4 里可用
  • fn 关键字开头
  • 只能包含 一个 表达式, 即返回表达式
  • return 关键字可忽略
  • 参数和返回类型均可做类型暗示

上面示例更严格的类型限定写法可写作:

$ids = array_map(fn(Post $post): int => $post->id, $posts);

有两点需要提及:

  • 还允许使用扩展操作符
  • 允许引用,两个参数都可以作为返回值

假如你想要通过引用的方式返回结果,应该使用以下语法:

fn&($x) => $x

简而言之,除了只允许一个表达式以外,简短的闭包和普通闭包的功能是一样的。

单行

你应该正确的理解它:短闭包只能有一个表达式。这意味着闭包体中不能有多行。

原因如下:短闭包的目的是为了减少冗余。当然,在任何情况下, fn 都比 function 短。然而, RFC 的创建者 Nikita Popov 认为,如果你要处理的是多行表达式的函数,那么使用闭包获得的益处就更少了。

毕竟,多行闭包的定义已经很冗余了,所以,有和没有这2个关键字( functionreturn )将不会有太大区别。

你是否同意这个观点取决于你自己。虽然我可以在我的项目中想到很多单行闭包的场景,但也有很多多行闭包的情况,从个人角度,我会喜欢这些情况下的简短语法。

不过还是有希望的:未来可能会添加多行短闭包,但那也是一个单独的 RFC 。

外部作用域的值

短闭包和普通闭包的另一个显著特征是,短闭包不需要用 use 关键字就能访问外部作用域的数据。

$modifier = 5;

array_map(fn($x) => $x * $modifier, $numbers);

需要注意的是,不能修改外部范围中的变量。因为它属于值传递而不是引用传递。这意味着你可以改变短闭包内的 $modifier 变量,但它不会对外部作用域中的 $modifier 变量产生影响。

当然,有一个例外,那就是 $this 关键字,它的作用与普通闭包中的作用完全相同:

array_map(fn($x) => $x * $this->modifier, $numbers);

发展前景

我已经提到过的多行闭包仍然是将来的一个发展可能.另外一个在我脑海中的想法就是在允许在类中使用短闭包,比如 getterssetters 函数.

class Post {
    private $title;

    fn getTitle() => $this->title;
}

总而言之,短闭包是一个很受欢迎的特性,尽管有很多地方需要提高.其中最有可能就是多行闭包了.

本文中的所有译文仅用于学习和交流目的,转载请务必注明文章译者、出处、和本文链接
我们的翻译工作遵照 CC 协议,如果我们的工作有侵犯到您的权益,请及时联系我们。

原文地址:https://stitcher.io/blog/short-closures-...

译文地址:https://learnku.com/php/t/28219

本帖已被设为精华帖!
本文为协同翻译文章,如您发现瑕疵请点击「改进」按钮提交优化建议
讨论数量: 7
suhanyujie

哈 这下好了,用新版的PHP写代码(箭头函数)可能会有种写Rust的感觉

4年前 评论
pub fn into_keypair(self) -> io::Result<Keypair> {
        use NodeKeyConfig::*;
        match self {
            Secp256k1(Secret::New) =>
                Ok(Keypair::generate_secp256k1()),

            Secp256k1(Secret::Input(k)) =>
                Ok(Keypair::Secp256k1(k.into())),

            Secp256k1(Secret::File(f)) =>
                get_secret(f,
                    |mut b| secp256k1::SecretKey::from_bytes(&mut b),
                    secp256k1::SecretKey::generate)
                .map(secp256k1::Keypair::from)
                .map(Keypair::Secp256k1),

            Ed25519(Secret::New) =>
                Ok(Keypair::generate_ed25519()),

            Ed25519(Secret::Input(k)) =>
                Ok(Keypair::Ed25519(k.into())),

            Ed25519(Secret::File(f)) =>
                get_secret(f,
                    |mut b| ed25519::SecretKey::from_bytes(&mut b),
                    ed25519::SecretKey::generate)
                .map(ed25519::Keypair::from)
                .map(Keypair::Ed25519),
        }
    }
4年前 评论

js 箭头函数主要是解决的匿名函数不会生成this,而是使用的上下文中的this。PHP这个解决了什么,难道只是增加了一个语法糖?

4年前 评论

看着像语法糖,啥时候出来了翻下代码看看确认下。

4年前 评论
zxk

rust 没有箭头函数啊rust是 |a| a+1, rust 的 match 表达式有箭头 match $expr { $expr=> $expr }

4年前 评论
zxk

php 的箭头函数,不需要use($a,...) ,可以自动捕获环境变量,这一点倒是跟 rust 很像

4年前 评论

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!