点击加入:
商务合作:请加微信(QQ):2230304070
精选文章正文
手册教程推荐:【】
免费领取
PHP 团队于2020年11月26日宣布 PHP 8 正式发布!

php8的下载地址:
PHP 8.0.0 目前是最新的主要版本,它引入了一些重大变更,以及许多新特性和性能优化,PHP 8.0 值得关注的改进包括以下:
1、PHP 8.0 引入了备受期待的 Just In Time (JIT) 编译器,能够进一步提高 PHP 脚本的执行速度
2、PHP 8.0 合并了诸多性能优化
3、JSON 支持现在被视为语言的核心部分,始终可用,而不是作为可选模块
4、支持 named 参数,因为它们能够指定参数名称而不是其确切顺序
5、支持类/属性/函数/方法/参数/常量的结构化元数据的属性(或在其他语言中也称为注释或修饰符)
6、支持可以指示多种不同类型的联合类型,这些类型可以用作参数或函数的返回类型
7、支持静态返回类型
8、str_contains()函数是一种检查字符串是否包含在另一个字符串中的简便方法,而不必使用strpos等。与之相似的是新的str_starts_with()和str_ends_with()函数
9、添加了Nullsafe运算符,作为在方法上应用空合并行为的快速简便的方法
10、相比较 PHP 7.4 稳定版,PHP 8.0 在性能上大约改进了 10%,但是至少在某些方面,JIT 可以提供更多的性能。
下面我们来看看新特性和性能优化
新增 ValueError 异常
这是8新引入进来的 ValueError 的内置异常类,它继承自 Exception 基类。你每次传递值到函数时候,如果检测到是一个无效的类型时抛出该异常,在 PHP 8 之前,这样的操作会直接做警告处理。
示例代码:
<?php declare(strict_types=1);
/** *?传递数组到?array_rand,类型正确,但是?array_rand?期望传入的是非空数组 *?所以会抛出?ValueError?异常 */ array_rand([],?0);
/** *?json_decode?的深度参数必须是有效的正整型值, *?所以这里也会抛出?ValueError?异常 */ json_decode('{}',?true,?-1);
运行结果:

新增对联合类型的支持
8新增的联合类型PHP运算符,它允许一个变量拥有多个类型的值。
示例代码如下:
<?php declare(strict_types=1);
/** *?定义一个支持联合类型的?Number?类 */ class?Number?{ ????private?int|float?$number;
????public?function?setNumber(int|float?$number):?void?{ ????????$this->number?=?$number; ????}
????public?function?getNumber():?int|float?{ ????????return?$this->number; ????} }
/** *?我们可以传递浮点型和整型值到?Number?对象 */ $number?=?new?Number(); $number->setNumber(5); var_dump($number->getNumber());
$number->setNumber(11.54); var_dump($number->getNumber());
exit;
运行结果:
重写方法时允许可变参数
当你在子类重写父类方法时,任何数量的参数都可以被替换成可变参数的,只要对应参数类型是兼容的就可以。
示例代码如下
<?php declare(strict_types=1);
class?A?{ ????public?function?method(int?$many,?string?$parameters,?$here)?{
????} } class?B?extends?A?{ ????public?function?method(...$everything)?{ ????????var_dump($everything); ????} }
$b?=?new?B(); $b->method('i?can?be?overwritten!'); exit;
运行结果:

静态返回类型
8 中可以使用 static 关键字标识某个方法,且返回该方法当前所属的类,即使它是继承的,可用于后期静态绑定。示例代码如下:
<?php declare(strict_types=1);
class?Test?{ ????public?function?doWhatever():?static?{ ????????//?Do?whatever. ????????return?$this; ????} }
exit;
新增 WeakMap 特性
WeakMap 允许你创建对象到任意值的映射(这个就类似 SplObjectStorage)的同时也不会阻止作为键的对象被垃圾回收。要是某个对象键被垃圾回收了,对应键值对就会从集合中被移除。
这一新特性非常有用,开发者不必担心代码存在内存泄露了。大多数 PHP 开发者可能对此不关心,但是当你在编写长时间运行的进程时,那你就一定要提防这个问题了,比如使用 ReactPHP 进行事件驱动编程时。用了 WeakMap 后引用的对象,就会在失效时自动被垃圾回收。
如果你在数组中做同样的操作,仍然会持有该对象的引用的,但是会导致内存泄露。
示例代码如下:
<?php declare(strict_types=1);
class?FooBar?{ ????public?WeakMap?$cache;
????public?function?__construct()?{ ????????$this->cache?=?new?WeakMap(); ????}
????public?function?getSomethingWithCaching(object?$obj)?{ ????????return?$this->cache[$obj]???=?$this->computeSomethingExpensive($obj); ????}
????public?function?computeSomethingExpensive(object?$obj)?{ ????????var_dump("I?got?called"); ????????return?rand(1,?100); ????} }
$cacheObject?=?new?stdClass;
$obj?=?new?FooBar;
//?"I?got?called"?只会打印一次 $obj->getSomethingWithCaching($cacheObject); $obj->getSomethingWithCaching($cacheObject);
var_dump(count($obj->cache));
//?删除该对象后?WeakMap?会释放相应内存 unset($cacheObject);
var_dump(count($obj->cache));
exit;
对应的运行结果:

变量语法调整
8的new 和 instanceof 关键字支持用于任意表达式了,示例代码如下
<?php declare(strict_types=1);
class?Foo?{} class?Bar?{}
$names?=?['Foo',?'Bar']; $class?=?new?($names[array_rand($names)]);
var_dump($class);
exit;
运行结果:

对象的类名字面量
8 中支持使用 $object::class 获取对象的类名,返回结果和 get_class($object) 是一样的。示例代码:
<?php declare(strict_types=1);
class?Test?{
}
$test?=?new?Test();
var_dump($test::class); var_dump(get_class($test)); exit;
运行结果:
参数列表中允许出现可选的尾部逗号
和数组中的尾部逗号一样,8也支持在参数列表中定义一个尾部逗号了。示例代码:
<?php declare(strict_types=1);
function?method_with_many_arguments( ????$a, ????$b, ????$c, ????$d, )?{ ????var_dump("this?is?valid?syntax"); }
method_with_many_arguments( ????1, ????2, ????3, ????4, );
exit;
上述代码运行结果:
Stringable 接口
8 引入了新的 Stringable 接口,只要某个类实现了 __toString 方法,就会被当作自动实现了 Stringable 接口(这一点和 Go 接口实现有些像),而不需要显式与声明实现该接口,示例代码:
<?php declare(strict_types=1);
class?Foo?{ ????public?function?__toString()?{ ????????return?'I?am?a?class'; ????} }
$obj?=?new?Foo; var_dump($obj?instanceof?Stringable);
exit;
运行结果:
throw 已经支持被用作表达式
8支持 throw 语句可以用在只允许表达式出现的地方,比如箭头函数、合并运算符和三元运算符等:
示例代码
<?php declare(strict_types=1);
$callable?=?fn()?=>?throw?new?Exception();
$nullableValue?=?null;
//?$value?是非空的 $value?=?$nullableValue????throw?new?\InvalidArgumentException();
exit;
捕获异常而不存储到变量
8可以编写 catch (Exception) 代码来捕获异常,但是不用将其存储到一个变量里:
<?php declare(strict_types=1);
$nullableValue?=?null;
try?{ ????$value?=?$nullableValue????throw?new?\InvalidArgumentException(); }?catch?(\InvalidArgumentException)?{ ????var_dump("Something?went?wrong"); }
exit;

上述代码运行结果:
PHP8的新增对注解的支持
注解实际上包含了多个 RFC:
https://wiki.php.net/rfc/attributes_v2 https://wiki.php.net/rfc/attribute_amendments https://wiki.php.net/rfc/shorter_attribute_syntax https://wiki.php.net/rfc/shorter_attribute_syntax_change
注解是 PHP 8 引入的最大新特性之一,一开始理解起来可能有点困难(如果你有 Java 基础的话理解起来会相对简单)。
注解允许你添加元数据到 PHP 函数、参数、类等,这些元数据随后就可以通过可编程方式获取到,在 PHP 7 或者更低版本中实现这样的功能需要解析代码注释块,而通过注解可以直接访问深度集成到 PHP 自身。
编写一段示例代码方便你理解,假设你想要允许开发者添加中间件到控制器类/方法,使用注解,你可以这么做,示例代码:
<?php declare(strict_types=1);
//?首先,我们需要定义注解,注解本身只是一个原生的?PHP?类,并且自身被打上了注解的注释
#[Attribute] class?ApplyMiddleware { ????public?array?$middlware?=?[];
????public?function?__construct(...$middleware) ????{ ????????$this->middleware?=?$middleware; ????} }
//?下面的语法会添加上述注解到?MyController?类,并且传入?auth?作为参数
#[ApplyMiddleware('auth')] class?MyController { ????public?function?index() ????{ ????} }
//?然后我们就可以在类中使用反射获取所有的?ApplyMiddleware?注解并读取给定的中间件参数
$reflectionClass?=?new?ReflectionClass(MyController::class);
$attributes?=?$reflectionClass->getAttributes(ApplyMiddleware::class);
foreach?($attributes?as?$attribute)?{ ????$middlewareAttribute?=?$attribute->newInstance(); ????var_dump($middlewareAttribute->middleware); }
exit;
运行上述代码,打印结果:

8新增构造函数属性提示支持
这个新特性是一个语法简写,支持将属性声明和构造函数属性初始化合并在一起,示例代码如下:
<?php declare(strict_types=1);
class?User?{ ????public?function?__construct( ????????public?int?$id, ????????public?string?$name, ????)?{} }
$user?=?new?User(1,?'Marcel');
var_dump($user->id); var_dump($user->name);
exit;
上述代码运行结果:
php8的Trait 支持定义抽象私有方法,示例代码如下:
<?php declare(strict_types=1);
trait?MyTrait?{ ????abstract?private?function?neededByTheTrait():?string;
????public?function?doSomething()?{ ????????return?strlen($this->neededByTheTrait()); ????} }
class?TraitUser?{ ????use?MyTrait;
????//?支持该语法 ????private?function?neededByTheTrait():?string?{?}
????//?不支持该语法?(错误的返回类型) ????//?private?function?neededByTheTrait():?stdClass?{?}
????//?支持该语法?(非静态方法变成了静态方法) ????//?private?static?function?neededByTheTrait():?string?{?} }
exit;
php8新增对 match 表达式支持
match 表达式和 switch 分支语句类似,不过在语义上match表达式会更加安全并且可以直接返回值:示例代码如下
<?php declare(strict_types=1);
echo?match?(1)?{ ????0?=>?'Foo', ????1?=>?'Bar', ????2?=>?'Baz', };
exit;
上述代码运行结果:
PHP 8引入了新的名为 mixed 的类型
该类型等价于 array| bool| callable |int |float |null |object |resource |string,示例代码如下:
<?php declare(strict_types=1);
function?debug_function(mixed?...$data) { ????var_dump($data); }
debug_function(1,?'string',?[]);
exit;
上述代码运行结果:

8新增对 命名参数 的支持
命名参数允许基于参数名称传递参数到函数,而不是参数所在的位置。那么这样一来,函数参数就可以自解释了且与顺序无关,并且允许跳过默认值,示例代码如下:
<?php declare(strict_types=1);
array_fill(start_index:?0,?num:?100,?value:?50);
exit;
新增对空安全运算符 ?-> 的支持
该运算符的左侧评估为 null 时,整个代码链路的执行就会被终止并且整体评估为 null。但是如果要不为 null ,那就要和普通的 -> 运算符功能一样:
<?php declare(strict_types=1);
class?User?{ ????public?function?getAddress()?{} }
$user?=?new?User();
$country?=?$user?->getAddress()?->country?->iso_code;
var_dump($country);
exit;
上述代码运行结果:
以上是本文的全部内容,希望对大家的学习有帮助,也希望大家多多支持php自学中心,学习与交流少不了一个圈子,点击加技术群:

(编辑:威海站长网)
【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!
|