太空船操作符
太空船操作符用于比较两个表达式。例如,当$a小于、等于或大于$b时,它分别返回-1、0或1。比较原则沿用PHP的常规比较规则进行。
<?php
//整数
echo 1 <=> 1; //0
echo 1 <=> 2; //-1
echo 2 <=>1; //1
标量类型声明和返回值的类型声明
PHP7可以对下面几种类型的参数做声明:字符串(string)、整型(int)、浮点型(float)以及布尔型(bool)。注意参数类型声明不受制于默认模式和严格模式。默认模式下、当传入的参数不符合声明类型时,会首先尝试转换类型;而严格模式下,则直接报错。(注意:这里的类型转换仅受制于可转换的类型,例如不能把’a‘转为int型)
参数类型做限制时:
#declare(strict_types=1);
function func(int $a) {
return $a;
}
var_dump(func(1));
var_dump(func('1'));
var_dump(func('a'));
//运行结果
//int(1)
//int(1)
//Fatal error: Uncaught TypeError: Argument 1 passed to sum() must be of the type int, string given
declare(strict_types=1);
function func(int $a) {
return $a;
}
var_dump(func(1));
var_dump(func('1'));
var_dump(func('a'));
//运行结果
//int(1)
//Fatal error: Uncaught TypeError: Argument 1 passed to sum() must be of the type int, string given,
返回值类型做限制时,其原理跟上述一致。如果返回的类型不是int型,在默认模式下,PHP会首先尝试转换返回值的类型为int型,如果不能转换,则会直接报错:
#declare(strict_types=1);
function func(int $a) : int {
return $a;
}
var_dump(func(1));
var_dump(func('1'));
var_dump(func('a'));
//运行结果
//int(1)
//int(1)
//Fatal error: Uncaught TypeError: Argument 1 passed to sum() must be of the type int, string given
declare(strict_types=1);
function func(int $a) : int {
return $a;
}
var_dump(func(1));
var_dump(func('1'));
var_dump(func('a'));
//运行结果
//int(1)
//Fatal error: Uncaught TypeError: Argument 1 passed to sum() must be of the type int, string given,
PHP7.1对函数返回值的声明做了补充,可以定义其返回值为void,无论是否开启严格模式,只要函数中有”return;“以外的其他return语句都会报错。
function func(int $a) : void {
//return $a; //Fatal error: A void function must not return a value
//return null; //Fatal error: A void function must not return a value (did you mean "return;" instead of "return null;"?)
return;
}
PHP7.1.0对参数类型和返回值类型还有进一步的支持,其类型可以是可空类型,在参数或返回值类型声明前边加上”?“,表示返回值要么是null,要么是声明的类型:
function func(?int $a): ?int {
return $a;
}
var_dump(func(null));
var_dump(func(1));
var_dump(func('a'));
//运行结果:
//NULL
//int(1)
//Fatal error: Uncaught TypeError: Argument 1 passed to func() must be of the type int or null, string given,
null合并操作符
在PHP7之前,人们经常会写这样的代码:
$page = isset($_GET['page']) ? $_GET['page'] : 0;
PHP7提供了一个新的语法”??“,如果变量存在且值不为null,它会返回自身的值,否则返回它的第二个操作数。
$page = $_GET['page'] ?? 0;
//如果有连续的三元运算符
$page = $_GET['page'] ?? $_POST['page'] ?? 0;
常量数组
在PHP7之前是无法通过define来定义一个数组常量的,PHP7支持了这个操作:
define('ANIMALS', ['dog', 'cat', 'bird']);
namespace批量导入
在PHP7之前,如果想要导入一个namespace下的多个class,我们需要这样写:
<?php
use Space\ClassA;
use Space\ClassB;
use Space\ClassC;
//PHP7中支持批量导入:
use Space\{ClassA, ClassB, ClassC}
throwable接口
在PHP7之前,如果代码中有语法错误,或者fatal error时,程序会直接报错退出,但是在PHP7中有了改变。PHP7实现了全局的throwable接口,原来的Exception和部分Error实现了该接口。这种Error可以像Exception一样被第一个匹配的try/catch捕获。如果没有匹配 的catch块,则调用异常处理函数进行处理。如果尚未注册异常处理函数,则按照传统方式处理(fatal error)
Error类并非继承自Exception类,所以不能用catch(Exception $e){…}来捕获Error。可以用catch(Error $e){…},或者通过注册异常处理函数(set_exception_handler())来捕获Error:
try {
undefindfunc();
}
catch(Error $e) {
var_dump($e);
}
//或者
set_exception_handler(function($e){
var_dump($e);
})
undefindfunc();
Closure::call()
在PHP7之前,我们需要动态地给一个对象添加方法时,可以通过Closure来复制一个闭包对象,并绑定到一个$this对象和类作用域:
<?php
class Test {
private $num = 1;
}
$f = function() {
return $this->num + 1;
};
$test = $f->bindTo(new Test, 'Test');
echo $test();
PHP7中新添加了Closure::call(),可以通过call来暂时绑定一个闭包对象到$this对象并调用它:
class Test{
private $num = 1;
}
$f = function() {
return $this->num + 1;
};
echo $f->call(new Test);
intdiv函数
PHP还新增了一个新的整除函数,在代码中不需要再手动转了:
//var_dump(intval(10 / 3));
var_dump(intdiv(10 , 3));
list的方括号写法
//之前
$arr = [1, 2, 3];
list($a, $b, $c) = $arr;
//PHP7.1.0
$arr = [1, 2, 3];
[$a, $b, $c] = $arr; //这里的[]并不是数组的意思,只是list的简略形式
除了上文这些,PHP还有很多其他的特性。例如,foreach遍历数组时不再修改内部指针,移除了ASP和script PHP标签、移除了$HTTP_RAW_POST_DATA、匿名类、类常量可见性等。读者可自行了解。