1、... in PHP 5.6+
5.6+之后,函数的参数列表可能包含...标记以表示该函数接受可变数量的参数。参数将作为数组传递给给定的变量
In PHP 5.6 and later, argument lists may include the ... token to denote that the function accepts a variable number of arguments. The arguments will be passed into the given variable as an array
function sum(...$numbers) {
$acc = 0;
foreach ($numbers as $n) {
$acc += $n;
}
return $acc;
}
echo sum(1, 2, 3, 4);
//output: 10
复制代码
2、标量类型声明
标量类型声明 有两种模式: 强制 (默认) 和 严格模式。 现在可以使用下列类型参数(无论用强制模式还是严格模式): 字符串(string), 整数 (int), 浮点数 (float), 以及布尔值 (bool)。它们扩充了PHP5中引入的其他类型:类名,接口,数组和 回调类型。
//...为php5.6+的
function sumOfInts(int ...$ints)
{
return array_sum($ints);
}
var_dump(sumOfInts(2, '3', 4.1));
//output: 9.1
复制代码
3、返回值类型声明
PHP 7 增加了对返回类型声明的支持。 类似于参数类型声明,返回类型声明指明了函数返回值的类型。可用的类型与参数声明中可用的类型相同。
function arraysSum(array ...$arrays): array
{
return array_map(function(array $array): int {
return array_sum($array);
}, $arrays);
}
print_r(arraysSum([1,2,3], [4,5,6], [7,8,9]));
//output
Array
(
[0] => 6
[1] => 15
[2] => 24
)
复制代码
4、null合并运算符??
由于日常使用中存在大量同时使用三元表达式和 isset()的情况, 我们添加了null合并运算符 (??) 这个语法糖。如果变量存在且值不为NULL, 它就会返回自身的值,否则返回它的第二个操作数。
// Fetches the value of $_GET['user'] and returns 'nobody'
// if it does not exist.
$username = $_GET['user'] ?? 'nobody';
// This is equivalent to:
$username = isset($_GET['user']) ? $_GET['user'] : 'nobody';
// Coalesces can be chained: this will return the first
// defined value out of $_GET['user'], $_POST['user'], and
// 'nobody'.
$username = $_GET['user'] ?? $_POST['user'] ?? 'nobody';
复制代码
5、太空船操作符(组合比较符)<=>
太空船操作符用于比较两个表达式。当
b时它分别返回-1、0或1。 比较的原则是沿用 PHP 的常规比较规则进行的。
// 整数
echo 1 <=> 1; // 0
echo 1 <=> 2; // -1
echo 2 <=> 1; // 1
// 浮点数
echo 1.5 <=> 1.5; // 0
echo 1.5 <=> 2.5; // -1
echo 2.5 <=> 1.5; // 1
// 字符串
echo "a" <=> "a"; // 0
echo "a" <=> "b"; // -1
echo "b" <=> "a"; // 1
复制代码
6、通过 define()
定义常量数组
Array
类型的常量现在可以通过define()
来定义。在PHP5.6
中仅能通过const
定义。
define('ANIMALS', [
'dog',
'cat',
'bird'
]);
echo ANIMALS[1]; // 输出 "cat"
复制代码
7、匿名类
现在支持通过
new class
来实例化一个匿名类,这可以用来替代一些“用后即焚”的完整类定义
interface Logger {
public function log(string $msg);
}
class Application {
private $logger;
public function getLogger(): Logger {
return $this->logger;
}
public function setLogger(Logger $logger) {
$this->logger = $logger;
}
}
$app = new Application;
$app->setLogger(new class implements Logger {
public function log(string $msg) {
echo $msg;
}
});
var_dump($app->getLogger());
//output
object(class@anonymous)#2 (0) {
}
复制代码
8、Unicode codepoint
转译语法
这接受一个以16进制形式的
Unicode codepoint
,并打印出一个双引号或heredoc
包围的UTF-8
编码格式的字符串。 可以接受任何有效的codepoint
,并且开头的0
是可以省略的。
echo "\u{aa}"; //ª
echo "\u{0000aa}";//ª
echo "\u{9999}";//香
//output:ªª香
复制代码
9、Closure::call()
Closure::call()
现在有着更好的性能,简短干练的暂时绑定一个方法到对象上闭包并调用它。
class A {private $x = 1;}
// PHP 7 之前版本的代码
$getXCB = function() {return $this->x;};
$getX = $getXCB->bindTo(new A, 'A'); // 中间层闭包
echo $getX();
// PHP 7+ 及更高版本的代码
$getX = function() {return $this->x;};
echo $getX->call(new A);
//output: 1 1
复制代码
10、为unserialize()
提供过滤
这个特性旨在提供更安全的方式解包不可靠的数据。它通过白名单的方式来防止潜在的代码注入。
// 将所有的对象都转换为 __PHP_Incomplete_Class 对象
$data = unserialize($foo, ["allowed_classes" => false]);
// 将除 MyClass 和 MyClass2 之外的所有对象都转换为 __PHP_Incomplete_Class 对象
$data = unserialize($foo, ["allowed_classes" => ["MyClass", "MyClass2"]);
// 默认情况下所有的类都是可接受的,等同于省略第二个参数
$data = unserialize($foo, ["allowed_classes" => true]);
复制代码
11、IntlChar
新增加的
IntlChar
类旨在暴露出更多的ICU
功能。这个类自身定义了许多静态方法用于操作多字符集的unicode
字符。
- 若要使用此类,请先安装
Intl
扩展
printf('%x', IntlChar::CODEPOINT_MAX);
echo IntlChar::charName('@');
var_dump(IntlChar::ispunct('!'));
//output
10ffff
COMMERCIAL AT
bool(true)
复制代码
12、整数除法函数 intdiv()
新加的函数
intdiv()
用来进行 整数的除法运算。
var_dump(intdiv(10, 3));// 3
复制代码
附件:PHP 类型比较表
- 在使用这些表格之前,需要明白变量类型及它们的意义。例如,
"42"
是一个字符串而42
是一个整数。FALSE
是一个布尔值而"false"
是一个字符串。 -
HTML 表单并不传递整数、浮点数或者布尔值,它们只传递字符串。要想检测一个字符串是不是数字,可以使用
is_numeric()
函数。 -
在没有定义变量
$x
的时候,诸如if ($x)
的用法会导致一个E_NOTICE
级别的错误。所以,可以考虑用 empty() 或者 isset() 函数来初始化变量。 - php.net/manual/zh/t…