我开始了解PHP 7.4 中的箭头函数。我试着像使用它们一样
<?php
$num = 1;
$arrowfunction = () => {
return $num + 1;
}
echo $arrowfunction();
因为我=>
在拉取请求中看到了运营商。就像javascript一样。
我期望'2'作为输出,但这不起作用!我有
解析错误:语法错误,第 3 行 /test.php 中的意外 ')'
我开始了解PHP 7.4 中的箭头函数。我试着像使用它们一样
<?php
$num = 1;
$arrowfunction = () => {
return $num + 1;
}
echo $arrowfunction();
因为我=>
在拉取请求中看到了运营商。就像javascript一样。
我期望'2'作为输出,但这不起作用!我有
解析错误:语法错误,第 3 行 /test.php 中的意外 ')'
PHP 中的箭头函数是在 PHP 7.4 中引入的。他们有点不同。
新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,它们支持?type
in 参数,这也允许参数为空。
$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 文件和命令行进行了测试,以检查一切是否正常。
在某些情况下,箭头函数可以使您的代码更短且更具可读性。它们的设计初衷是考虑将它们用于简单的回调。例如,考虑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";
}
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]));
减少不必要的语法有助于理解代码的真正目的,但请记住,更短并不总是意味着更简洁!我建议像对待三元运算符一样谨慎对待箭头函数。仅当您知道它们有助于提高可读性时才使用它们,而不仅仅是为了使您的代码更短。