编程

PHP 8.5 发布在即!新特性一览

13 2025-11-14 10:04:00

PHP 8.5 将于本月,即 2025 年 11 月 20 日发布,其特色功能包括管道运算符、新的 URI 扩展、新的 array_first()array_last() 函数等等:

管道运算符

对于希望将多个可调用对象链接在一起,并以原生方式从左到右传递值的 PHP 开发人员来说,管道运算符是一个令人兴奋的工具:

// Using the pipe operator in PHP 8.5
$result = "Hello World"
    |> htmlentities(...)
    |> str_split(...)
    |> fn($x) => array_map(strtoupper(...), $x)
    |> fn($x) => array_filter($x, fn($v) => $v != 'O');

原生 array_first()array_last() 函数

PHP 8.5 将引入 array_first()array_last() 函数。虽然表面上看,这些函数似乎无关紧要(而且 PHP 社区已经有用户实现和 polyfill),但作为 PHP 语言的原生函数,它们早就应该出现了。

这些函数是对 PHP 7.3 中已合并的数组键方法的补充:

在 PHP 7.3 中,我们有了 array_key_first()array_key_last() 来获取数组的第一个和最后一个键。但我们目前还没有获取数组的第一个和最后一个值的方法。这比你想象的要难,因为:

  • 数组键不一定是整数,也不一定从 0 开始,等等……
  • reset()end() 这样的现有“技巧”在语义上是不正确的,因为它们会修改数组的“内部迭代器”。此外,它并非适用于所有类型的表达式(例如,函数返回的数组/普通数组可能会因为引用传递参数而触发警告)。
  • 使用 $array[array_key_first($array)] 比较繁琐。

如果想深入了解 array_first()array_last(),我们有一篇更详细的文章,但这里提供一些函数签名及其工作原理的示例:

// Function signatures
function array_first(array $array): mixed {}
function array_last(array $array): mixed {}
 
// Examples
array_first(["single element"]); // "single element"
array_last(["single element"]); // "single element"
 
array_first([]); // NULL
array_last([]); // NULL
 
array_first([1 => 'a', 0 => 'b', 3 => 'c', 2 => 'd']); // 'a'
array_last([1 => 'a', 0 => 'b', 3 => 'c', 2 => 'd']); // 'd'
 
$str = "hello";
array_first([&$str, false]); // "hello" (no ref)
array_last([false, &$str]); // "hello" (no ref)

新的 URI 扩展

PHP 8.5 引入了一个新的 URI 扩展,它是一个符合标准的解析器,“同时支持 RFC 3986 和 WHATWG URL 标准,并作为其标准库中始终可用的一部分,包含在一个新的 'URI' 扩展中。” RFC 中提供了大量示例,以下是 PHP 基金会公告文章中的一个示例,展示了 RFC 3986 的 URI 类:

use Uri\Rfc3986\Uri;
 
$url = new Uri('HTTPS://thephp.foundation:443/sp%6Fnsor/');
 
$defaultPortForScheme = match ($url->getScheme()) {
    'http' => 80,
    'https' => 443,
    'ssh' => 22,
    default => null,
};
 
// Remove default ports from URLs.
if ($url->getPort() === $defaultPortForScheme) {
    $url = $url->withPort(null);
}
 
// Getters normalize the URL by default. The `Raw`
// variants return the input unchanged.
 
echo $url->toString(), PHP_EOL;
// Prints: https://thephp.foundation/sponsor/
echo $url->toRawString(), PHP_EOL;
// Prints: HTTPS://thephp.foundation/sp%6Fnsor/

获取当前正在执行的闭包

PHP 8.5 将支持闭包中的递归,方法是获取当前正在执行的闭包(感谢 Alexandre Daubois 的贡献)。正如 rfc:closure_self_reference RFC 中指出的,目前的解决方法是将变量引用绑定到闭包中。PHP 8.5 提供了静态方法 Closure::getCurrent() 来获取当前正在运行的闭包:

$fibonacci = function (int $n) {
    if (0 === $n || 1 === $n) {
        return $n;
    }
 
    $fn = Closure::getCurrent();
 
    return $fn($n - 1) + $fn($n - 2);
};
 
echo $fibonacci(10) . "\n";

常量表达式中的闭包支持

PHP 8.5 引入了对常量表达式中闭包的支持,使得将默认属性值定义为闭包成为可能,以及其他一些用例:

function my_array_filter(
    array $array,
    Closure $callback = static function ($item) { return !empty($item); },
) {
    $result = [];
 
    foreach ($array as $item) {
        if ($callback($item)) {
            $result[] = $item;
        }
    }
 
    return $result;
}
 
my_array_filter([
    0, 1, 2,
    '', 'foo', 'bar',
]); // [1, 2, "foo", "bar"]

PHP 致命错误回溯

新增的 fatal_error_backtraces 设置可以控制是否显示致命错误的回溯信息。在 PHP 8.5 中,fatal_error_backtraces 设置的默认值为 1——你无需进行任何配置即可获取这些回溯信息(当然,你也可以根据需要禁用它们)。在当前稳定的 PHP 版本(例如 PHP 8.4)中,一些致命错误可能没有回溯信息,例如解析错误(语法错误)、重复的函数或类、无限循环且执行时间超过上限等。

Fatal error: Cannot redeclare class B (previously declared in /srv/app/index.php:11) in /srv/app/b.php on line 3
Stack trace:
#0 /srv/app/index.php(6): require()
#1 /srv/app/index.php(21): A->loadClassB()
#2 {main}

INI 差异选项

PHP 8.5 为 php --ini 标志引入了 INI 差异选项,使识别配置中已更改的 INI 值变得容易。`--ini` 标志有助于显示已加载的 php.ini 配置文件以及解析的其他 .ini 文件:

$ php --ini=diff
Non-default INI settings:
allow_url_include: "0" -> ""
auto_append_file: (none) -> ""
auto_prepend_file: (none) -> ""
display_errors: "1" -> ""
display_startup_errors: "1" -> ""
enable_dl: "1" -> ""
error_reporting: (none) -> "22527"
...

更多

 

下一篇