12

我开始了解PHP 7.4 中的箭头函数。我试着像使用它们一样

<?php
$num = 1;
$arrowfunction = () => {
   return $num + 1;
}
echo $arrowfunction();

因为我=>在拉取请求中看到了运营商。就像javascript一样。

我期望'2'作为输出,但这不起作用!我有

解析错误:语法错误,第 3 行 /test.php 中的意外 ')'

4

2 回答 2

32

PHP 中的箭头函数是在 PHP 7.4 中引入的。他们有点不同

fn 关键字

fn关键字现在是保留关键字

以前,我们曾经继续使用function关键字。

$add = function ($valone,$valtwo) {
    return $valone + $valtwo;
};
$add(1,2) // 3

随着新箭头函数的出现:

$add = fn($valone,$valtwo) => $valone + $valtwo;
$add(1,2) // 3

父范围

早些时候,我们必须遵循关键字的用法来use参与来自父范围的变量。

$y = 1;
$fn = function ($x) use ($y) {
    return $x + $y;
};
echo $fn(2); // 3

在父范围中定义的表达式将按值隐式捕获。

$y = 1;
$fn = fn($x) => $x + $y;
echo $fn(2); // 3

以上是针对$this类方法内部的变量。

class foo {
   public function test() {
       $context = fn() => var_dump($this);
       $context(); 
   }
}
$test = new foo();
$test->test();  // object(foo)#1 (0) { }

就像之前一样,我们过去通过使用use关键字从父作用域中获取变量来执行我们的操作,因此这意味着我们不能将函数中的变量值写入上层作用域。

$y = 1;
$fn = fn() => $y++;
$fn(); // Has no effect
echo $y  // 1

如果我们正在考虑从闭包中分配另一个变量的值,那么这也行不通

$y = 1;
$f = 0;
$fn = fn() => $f = $y + 1;
$fn();
echo $f; // 0

函数签名

这在 PHP 中是全新的,它允许我们定义函数的类型、变量和函数返回的值

fn(int $x) => $x; // the argument type must be (int)
fn(): int => $x; // type of return value (int)

如果在调用函数时未将定义的参数类型放在参数中,则会引发错误。可以使用TypeError类型捕获错误

$var = 10;
$int_fn = fn(int $x): int => $x;
var_dump($int_fn($var)); // int(10)
try {
    $int_fn("foo");
} catch (TypeError $e) {
    echo $e->getMessage(), "\n"; // Argument 1 passed to {closure}() must be of the type int, string given, called in x on line y
}

通过 PHP 7.1,它们支持?typein 参数,这也允许参数为空。

$funn = fn(?int... $args): array => $args;
var_dump($funn(20, null, 30)); // Array(3) { [0]=> int(20) [1]=> NULL [2]=> int(30) }

如果您向上述函数提供字符串或其他任何内容而不是 int,那么您将收到错误消息

传递给 {closure}() 的参数必须是 int 或 null 类型,给定字符串,在第 y 行的 x 中调用

嵌套箭头函数

$var = 6;
var_dump((fn() => fn() => $var)()());  // int(6)
var_dump((fn() => function() use($var) { return $var; })()()); // int(6)

除非调用,否则不会抛出闭包内的任何可能错误

ini_set('display_errors', 1);
ini_set('display_startup_errors', 1);
error_reporting(E_ALL);
$b = 1;
fn() => $b + $c; // no error, nothing


ini_set('display_errors', 1);
ini_set('display_startup_errors', 1);
error_reporting(E_ALL);
$b = 1;
(fn() => $b + $c)(); // Notice: Undefined variable: c in the location on line x

如果错误报告关闭,那么您将获得int(1)

如何使用 PHP。现在7.4?
为了快速在线测试,只需将这些代码粘贴

到您的本机系统,我刚刚克隆了 php-src 的这个分支并使用 GCC 和 make 编译它。我通过 test.php 文件和命令行进行了测试,以检查一切是否正常。

核心参考 - https://wiki.php.net/rfc/arrow_functions_v2

于 2019-06-18T21:41:47.663 回答
8

在某些情况下,箭头函数可以使您的代码更短且更具可读性。它们的设计初衷是考虑将它们用于简单的回调。例如,考虑usort()将回调函数作为用户参数。

在 PHP 7 之前,你必须做这样的事情来定义你自己的回调usort()

// old syntax prior to PHP 7
function cmp($a, $b) {
    if ($a == $b) {
        return 0;
    }
    return ($a < $b) ? -1 : 1;
}

$a = [3, 2, 5, 6, 1];

usort($a, "cmp");

foreach ($a as $key => $value) {
    echo "$key: $value\n";
}

PHP 7 添加了一个spaceship 运算符,现在由于箭头函数,您可以使您的代码更加简洁。

// New syntax since PHP 7.4
$a = [3, 2, 5, 6, 1];

usort($a, fn($a, $b) => $a<=>$b);

foreach ($a as $key => $value) {
    echo "$key: $value\n";
}

3v4l.org在线试用

PHP 中的匿名函数可能非常冗长,即使它们只执行简单的操作,这也是语法较短的原因。作为另一个示例,请考虑以下函数:

// Returns an array with each element squared - old syntax
function array_square($arr) {
    return array_map(function($x) { return $x*$x; }, $arr);
}

// Returns an array with each element squared - new syntax
function array_square($arr) {
    return array_map(fn($x) => $x**2, $arr);
}

print_r(array_square([1,2,3,4,5]));

减少不必要的语法有助于理解代码的真正目的,但请记住,更短并不总是意味着更简洁!我建议像对待三元运算符一样谨慎对待箭头函数。仅当您知道它们有助于提高可读性时才使用它们,而不仅仅是为了使您的代码更短。

于 2019-06-18T23:36:46.360 回答