PHP 7.2 新功能介绍

clipboard.png

PHP 7.2 已經在 2017 年 11 月 30 日 正式發布 。這次發布包含新特性、功能,及優化,以讓我們寫出更好的代碼。在這篇文章裡,我將會介紹一些 PHP 7.2 最有趣的語言特性。php

你能够在 Requests For Comments 頁面查看完整的更動清單。html

核心改进

参数类型声明

从 PHP5 起,咱们能够指定函数参数的预期声明类型。若是传参类型错误,PHP 就会抛出一个错误。laravel

参数类型声明 (也称类型提示) 指定预期要传参给函数或者类方法的参数类型。

这里有个例子:git

class MyClass {
    public $var = 'Hello World';
}

$myclass = new MyClass;

function test(MyClass $myclass){
    return $myclass->var;
}

echo test($myclass);

在这段代码中,测试函数须要一个 MyClass 实例。不正确的参数数据类型会致使一个致命错误。算法

Fatal error: Uncaught TypeError: Argument 1 passed to test() must be an instance of MyClass, string given, called in /app/index.php on line 12 and defined in /app/index.php:8

从 PHP 7.2 类型提示 能够被用在对象型数据上,而且这个改进容许通用对象类型做为一个函数或者方法的参数。这里有个例子:docker

class MyClass {
    public $var = '';
}

class FirstChild extends MyClass {
    public $var = 'My name is Jim';
}
class SecondChild extends MyClass {
    public $var = 'My name is John';
}

$firstchild = new FirstChild;
$secondchild = new SecondChild;

function test(object $arg) {
    return $arg->var;
}

echo test($firstchild);

echo test($secondchild);

在以上示例中,咱们调用了两次测试函数,每次都传递一个不一样的对象。这在以前的 PHP 版本中是史无前例的。编程

file

在 Docker 中测试 PHP 7.0 和 PHP 7.2 的类型提示。c#

对象返回类型声明

若变量类型指定函数参数的预期类型,返回值类型一样也能够被指定预期类型。数组

返回类型声明 指定一个函数应该返回的预期类型。

PHP 7.2 起,对象数据类型可使用返回类型声明。这里有个例子:安全

class MyClass {
    public $var = 'Hello World';
}

$myclass = new MyClass;

function test(MyClass $arg) : object {
    return $arg;
}

echo test($myclass)->var;

以前的 PHP 版本会抛出如下致命错误:

Fatal error: Uncaught TypeError: Return value of test() must be an instance of object, instance of MyClass returned in /app/index.php:10

固然,PHP 7.2 的代码会打印出 'Hello World'。

参数类型泛化

PHP 目前是不容许子类和它父类或者接口的参数类型有任何差别的。 这是什么意思呢?
参考下如下代码:

<?php
class MyClass {
    public function myFunction(array $myarray) { /* ... */ }
}

class MyChildClass extends MyClass {
    public function myFunction($myarray) { /* ... */ }
}

这里咱们省略了子类中的参数类型。 在 PHP 7.0 中,会产生如下警告:

Warning: Declaration of MyChildClass::myFunction($myarray) should be compatible with MyClass::myFunction(array $myarray) in %s on line 8

从 PHP 7.2 起,咱们能够忽略子类中的类型 而不会破坏任何代码。这个方案使得咱们能够在库中升级类,从而可使用类型提示,却无需更新其全部的子类。

列表语法中的尾随逗号

在 PHP 数组的最后一个元素上使用尾随逗号是 合法语法 ,而且 有时候鼓励这么作 ,能够很轻松的避免增长新元素的时候出现缺失逗号的错误。 从 PHP 7.2 在 分组命名空间 中,咱们可使用尾随逗号 。

参阅 列表语法中的尾随逗号 得到 RFC 的直观感知和一些示例代码。

安全性改进

密码哈希中的Argon2

Argon2 是荣获 2015 年密码哈希算法比赛中的冠军的强大哈希算法, PHP 7.2 将其做为安全  Bcrypt 算法的替代品。
新版的 PHP 中引入了 PASSWORD_ARGON2I 常量,如今能够在 password_* 系列函数中使用:

password_hash('password', PASSWORD_ARGON2I);

与只使用一个 cost 因子的 Bcrypt 不一样, Argon2 使用三个 cost 因子 区分以下:

  • 定义哈希计算期间应该消耗的KiB数量的内存开销(默认值为1 << 10或1024 KiB或1 MiB)
  • 定义哈希算法迭代次数的时间开销(默认值为2)
  • 并行因子,用于设置哈希计算时使用的并行线程数(缺省值为2)

如下三个新常量定义了默认的 cost 因子:

  • PASSWORD_ARGON2_DEFAULT_MEMORY_COST
  • PASSWORD_ARGON2_DEFAULT_TIME_COST
  • PASSWORD_ARGON2_DEFAULT_THREADS

这里有个例子:

$options = ['memory_cost' => 1<<11, 'time_cost' => 4, 'threads' => 2];
password_hash('password', PASSWORD_ARGON2I, $options);

查阅 Argon2 密码哈希 的更多信息。

Libsodium 成为 PHP 核心的组成部分

从 7.2 版开始,PHP 在其核心中涵盖了 Sodium library 。 Libsodium 是一个跨平台和跨语言的库,用于加密,解密,签名,密码哈希等。
这个库以前是 经过 PECL 来提供的。
有关 Libsodium 函数列表,参阅 快速入门
也可参阅 PHP 7.2: 第一个将现代加密技术添加到其标准库的编程语言

弃用

这里有个 PHP 7.2 弃用函数和特性 清单,PHP 8.0 以后将所有移除。

PHP 5.1 中 __autoload 函数已被 spl_autoload_register 取代。如今会在编译期间报一个弃用通知。

当抛出致命错误的时候,会建立 $php_errormsg 局部变量。 PHP 7.2 中应该使用 error_get_last 和 error_clear_last 替代这种作法。

create_function() 能够建立一个具备函数名称的函数,将函数参数和函数体做为该函数的列表传入。由于安全问题和性能表现不佳,它被标记为弃用,鼓励用封装替代。

mbstring.func_overload ini 设置为非零值已经被标记为弃用。

(unset) cast 是个老是返回 null 的表达式,而且毫无用处。

若是传入第二个参数,parse_str() 将查询字符串解析到数组当中, 不然解析到本地符号表。 由于安全缘由, 不建议 在函数做用域中动态设置变量,使用不带第二个参数的 parse_str() 将抛一个弃用通知。

gmp_random() 是平台相关的,将会被废弃。使用 gmp_random_bits() 和 gmp_random_rage() 代替。

each() 在数组上迭代的行为很是像 foreach(),但 foreach() 基于一些缘由而成为更优选择,例如它的速度快上 10 倍。如今在循环中使用前者将会抛出一个废弃提示。

 assert() 函数检查给定的断言,并在结果为 FALSE 的时候进行相关处理。 带有字符串参数的 assert() 如今已经弃用,由于它有 RCE 漏洞。 zend.assertion ini 选项能够关闭断言表达式。

$errcontext 是一个包含产生错误时的局部变量数组。它可被做为错误处理程序 set_error_handler() 函数的最后一个参数。

PHP 7.2 对 WordPress 用户意味着什么?

根据官方 WordPress 统计页 所示,截至撰写本文时,只有 19.8% 的 WordPress 用户升级到了 PHP 7。只有 5%使用 PHP 7.1。你能够看到超过 40% 的用户仍然使用 PHP 5.6,更可怕的是超过 39% 的用户在使用已经不受支持的 PHP 版本。截至 2016 年 12 月,WordPress.org 为 PHP 5.6 版本的用户修改 官方建议 为建议使用 PHP 7 或以上的版本。
WordPress PHP 7.1 stats

WordPress PHP 7.1 数据统计

以上的数据表现并不使人愉悦,由于看上去 PHP 7 好像更快点。下面是一些统计数据:

  • PHP 官方 基准测试 显示 PHP 7 容许系统每秒执行2次请求,与 PHP 5.6 相比,几乎只是通常的延迟。
  • Christian Vigh 也发布了一个 PHP 性能测试对比 他发现 PHP 5.2 比 PHP 7 慢了近 400%。

咱们在 2018 运行了性能基准测试 PHP 5.6 vs PHP 7 vs HHVM。与上述基准测试相似,咱们发现 PHP 7.2 与 PHP 5.6 相比每秒可执行几乎三倍数量的事务(请求)。

WordPress benchmarks

WordPress 基准测试

  • WordPress 4.9.4 PHP 5.6 基准测试结果: 49.18 req/sec
  • WordPress 4.9.4 PHP 7.0 基准测试结果: 133.55 req/sec
  • WordPress 4.9.4 PHP 7.1 基准测试结果:134.24 req/sec
  • WordPress 4.9.4 PHP 7.2 基准测试结果:148.80 req/sec ?
  • WordPress 4.9.4 HHVM 基准测试结果:144.76 req/sec

许多东西在仅仅在更新上比较慢,由于要花时间去参与测试全部新的第三方插件和主题确保它们能够正常运行。不少时候,慢是由于它们还没完成。不肯定你运行的 PHP 是什么版本?其中一个很最简单的方法就是使用这个工具  Pingdom  或者 Google Chrome开发工具.。第一个 HTTP 请求头通常将会展现你的版本。

Check version of PHP

检查 PHP 版本

这将依赖于主机不修改 X-Powered-By 头信息的值。若是修改了的话,你可能就看不到 PHP 的版本信息了,这种状况下你须要 经过 FTP 上传文件.。或者你老是去询问主机。

升级到 PHP 7.2

PHP 7.2 还有一部分没完成,可是你能够先尝尝鲜。你能够 测试你的 WordPress 本地站点 或者在相似 Docker 环境中检查你的脚本,你能够在命令行中测试比较不一样的 PHP 版本。

结语

准备好切换到 PHP 7.2 了吗?不过至少但愿你首先已通过渡到了 PHP 7 以上的版本了。若是你如今还没准备好测试的话,那么,升级你的脚本,检查你的代码,说说你对 PHP 7.2 的首次体验。

更多现代化 PHP 知识,请前往 Laravel / PHP 知识社区
相关文章
相关标签/搜索