Use PHP7

说明

目前RC3中,PHP 7.2计划于11月30日发布。新版本将提供新的特性,功能和改进,使我们能够编写更好的代码。在这篇文章中,我将介绍一些PHP 7.2中最有趣的语言特性。

参数类型声明

从PHP 5开始,我们可以在函数的声明中指定期望传递的参数类型。如果给定的值是不正确的类型,那么PHP会抛出一个错误。参数类型声明(也称为类型提示)指定预期要传递给函数或类方法的变量的类型。

例如下面这个例子:

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 类型提示可以与对象数据类型一起使用,并且这种改进允许声明通用对象作为函数或方法的参数。这里是一个例子:

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版本中这是不可能的。

Docker命令
在Docker中使用PHP 7.0和PHP 7.2测试类型提示

对象返回类型声明

如果参数类型声明指定函数参数的预期类型,则返回类型声明指定返回值的预期类型。

返回类型声明指定了一个函数应该返回的变量的类型。
从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);

与仅使用一个成本因素的Bcrypt不同,Argon2需要三个成本因素区分如下:

  • 甲存储器成本它定义了应散列期间被消耗KIB的数(默认值是1 << 10,或1024 KIB,或1 MIB)
  • 甲时间成本定义散列算法的迭代次数(默认为2)
  • 一个并行因子,用于设置散列期间将使用的并行线程数(缺省值为2)

三个新的常量定义了默认的成本因素:

  • 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 Core的一部分

从版本7.2开始,PHP将钠库纳入核心。Libsodium 是一个跨平台和跨语言的库,用于加密,解密,签名,密码散列等等。以前通过 PECL 提供。有关Libsodium功能的文档列表,请参阅库快速入门指南。另请参阅PHP 7.2:将现代加密技术添加到其标准库中的第一种编程语言。

弃用

以下是PHP 8.0 不推荐使用的函数和功能列表,不晚于PHP 8.0:

该__autoload功能已被取代由spl_autoload_register在PHP 5.1。现在,在编译期间遇到弃用通知。

在$ php_errormsg中时,抛出一个非致命错误变量是在局部范围内创建。由于应该使用PHP 7.2 error_get_lasterror_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倍。现在,在循环的第一个呼叫中将会抛弃。

所述断言函数检查给定的断言,并采取适当的行动,如果结果是FALSE。带有字符串参数的 assert() 的使用现在已被弃用,因为它会打开一个 RCE 漏洞。该 zend.assertion INI 选项可用于防止断言表达式的评价。

$ errcontext是包含生成错误时存在的局部变量的数组。它作为使用 set_error_handler() 函数设置的错误处理程序的最后一个参数传递。

结尾说明

翻译自 https://kinsta.com/blog/php-7-2/

Script Maker Day Day Up!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值