标量类型声明有两种模式:强制(默认)模式、严格模式。下列参数类型可以使用(无论用强制模式还是严格模式):字符串(string)、整形(int)、浮点数(float)和布尔型(bool)。其他类型在 PHP5 中有支持:类名、接口、数组和可被调用的。
<?php
// Coercive mode
function sumOfInts(int ...$ints)
{
return array_sum($ints);
}
var_dump(sumOfInts(2, '3', 4.1));
上述例子输出:
int(9)
当开启严格模式后,一个 declare 声明必须置于 PHP 脚本文件开头,这意味着严格声明标量是基于文件可配的。这个指令不仅影响参数的类型声明,也影响到函数的返回值声明(详见下面的返回值声明)。
详细的标量类型声明的文档与示例,可以查看类型声明页面。
PHP7 新增了返回类型声明,类似于参数类型声明,返回类型声明提前声明了函数返回值的类型。可用的声明类型与参数声明中可用的类型相同。
<?php
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]));
上述代码返回值为:
Array
(
[0] => 6
[1] => 15
[2] => 24
)
详细的返回值声明相关的文档和示例代码可以查阅返回值声明文档。
空合并算子的操作符为 ??
,已经作为一种语法糖用于日常需求中用于三元表达式,它与 isset() 同时发生。如果变量存在且不为空,它就会返回对应的值,相反,它返回它的第二个操作数。
<?php
// 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';
?>
太空船操作符(组合比较符,RFC)
太空船操作符用于比较两个表达式。它返回一个大于 0、等于 0、小于 0 的数,用于表示 $a 与 $b 之间的关系。比较的原则是沿用 PHP 的常规比较规则进行的。
<?php
// Integers
echo 1 <=> 1; // 0
echo 1 <=> 2; // -1
echo 2 <=> 1; // 1
// Floats
echo 1.5 <=> 1.5; // 0
echo 1.5 <=> 2.5; // -1
echo 2.5 <=> 1.5; // 1
// Strings
echo "a" <=> "a"; // 0
echo "a" <=> "b"; // -1
echo "b" <=> "a"; // 1
?>
Array 类型的常量可以通过 define() 来定义。在 PHP5.6 中仅能通过 const 定义。
<?php
define('ANIMALS', [
'dog',
'cat',
'bird'
]);
echo ANIMALS[1]; // outputs "cat"
?>
可以通过 new 关键字初始化一个匿名类。匿名类使用场景与完整的类场景相同。
<?php
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());
?>
上面代码输出:
object(class@anonymous)#2 (0) {
}
详细文档可以参考匿名类文档
通过十六进制内容与双引号组成的字符串生成 Unicode codepoint,可以接受任何有效的 codepoint,并且开头的 0 是可以省略的。
echo "\u{aa}";
echo "\u{0000aa}";
echo "\u{9999}";
上面代码输出:
(same as before but with optional leading 0's)
闭包 Closure::call() 有着更好的性能,简短干练的暂时绑定一个方法到对象上闭包并调用它。
<?php
class A {private $x = 1;}
// Pre PHP 7 code
$getXCB = function() {return $this->x;};
$getX = $getXCB->bindTo(new A, 'A'); // intermediate closure
echo $getX();
// PHP 7+ code
$getX = function() {return $this->x;};
echo $getX->call(new A);
上述代码输出:
1
1
这个特性意在提供更安全的方式解包不可靠的数据。通过白名单的方式来防止代码注入。
<?php
// converts all objects into __PHP_Incomplete_Class object
$data = unserialize($foo, ["allowed_classes" => false]);
// converts all objects into __PHP_Incomplete_Class object except those of MyClass and MyClass2
$data = unserialize($foo, ["allowed_classes" => ["MyClass", "MyClass2"]);
// default behaviour (same as omitting the second argument) that accepts all classes
$data = unserialize($foo, ["allowed_classes" => true]);
新增加的 IntlChar 类意在于暴露出更多的 ICU 功能。类自身定义了许多静态方法用于操作 unicode 字符。
<?php
printf('%x', IntlChar::CODEPOINT_MAX);
echo IntlChar::charName('@');
var_dump(IntlChar::ispunct('!'));
上述代码输出:
10ffff
COMMERCIAL AT
bool(true)
若要使用此类,请先安装Intl扩展。
预期(增强的断言)是向后兼用以增强 assert() 方法。在代码中启用断言为零成本,并且提供抛出特定异常的能力。
在使用老版本 API 时,如果第一个参数是一个字符串,那么它将被解析。第二个参数可以是一个简单的字符串(导致 AssertionError 被触发),或一个包含一个错误消息的自定义异常对象。
<?php
ini_set('assert.exception', 1);
class CustomError extends AssertionError {}
assert(false, new CustomError('Some error message'));
上述代码输出:
Fatal error: Uncaught CustomError: Some error message
这个特性会带来两个 PHP.ini 设置(以及它们的默认值):
- zend.assertions = 1
- assert.exception = 0
zend.assertions 有三种值:
- 1 = 生成并且执行代码(开发模式)
- 0 = 执行代码并且在运行期间跳来跳去
- -1 = 不生成任何代码 (0开销, 生产模式)
assert.exception 意味着断言失败时抛出异常。默认关闭保持兼容旧的 assert() 函数。
在 PHP7 之前需要声明一大堆命名空间,但是现在可以通过 use 的新特性,批量声明。
<?php
// Pre PHP 7 code
use some\namespace\ClassA;
use some\namespace\ClassB;
use some\namespace\ClassC as C;
use function some\namespace\fn_a;
use function some\namespace\fn_b;
use function some\namespace\fn_c;
use const some\namespace\ConstA;
use const some\namespace\ConstB;
use const some\namespace\ConstC;
// PHP 7+ code
use some\namespace\{ClassA, ClassB, ClassC as C};
use function some\namespace\{fn_a, fn_b, fn_c};
use const some\namespace\{ConstA, ConstB, ConstC};
这个特性是建立在 PHP 5.5 的生成器特性之上的。现在允许在生成器中使用 return 语句提供一个最终的返回值(返回引用是不允许的)。这个值可以通过新的方法获取:Generator::getReturn(),但这个方法只能在生成器迭代完成后调用。
<?php
$gen = (function() {
yield 1;
yield 2;
return 3;
})();
foreach ($gen as $val) {
echo $val, PHP_EOL;
}
echo $gen->getReturn(), PHP_EOL;
上述代码输出:
1
2
3
给生成器提供一个最终的返回值是一个很有用的特性。因为最终的返回值(或许来自某种形式的协程计算)便于在使用生成器的时候进行特殊处理。这比之前需要不停的去检查是否已经迭代到最终值要方便多了。
生成器可以委托到另一个生成器中。现在允许生成器使用新的 yield from 语法进入任何实现 Traversable 接口的对象或数组中进行迭代。这个特性使得迭代可以被分解成多个操作,从而编写出重用性更高、更干净的代码。
<?php
function gen()
{
yield 1;
yield 2;
yield from gen2();
}
function gen2()
{
yield 3;
yield 4;
}
foreach (gen() as $val)
{
echo $val, PHP_EOL;
}
上述代码输出:
1
2
3
4
intdiv() 函数来处理整除,并返回一个整数。
<?php
var_dump(intdiv(10, 3));
上述代码输出:
int(3)
该特性给 session_start() 函数提供一些设置能力,当然这些设置可以在 PHP.ini 中设置。
<?php
session_start(['cache_limiter' => 'private']); // sets the session.cache_limiter option to private
这个特性还引入了一个新的 php.ini 设置( session.lazy_write ),默认情况下为 true,表示改变会话数据只是重写。
这个新功能,当你使用 preg_replace_callback() 函数时代码更清晰可读。在PHP7之前,每个正则表达式都需要回调( preg_replace_callback() 函数的第二个参数 )中来实现功能,这会使流程混乱不可控。
现在,回调可以通过与正则表达式绑定着写,只需将正则表达式作为 key,回调函数作为 value。
<?php
$tokenStream = []; // [tokenName, lexeme] pairs
$input = <<<'end'
$a = 3; // variable initialisation
end;
// Pre PHP 7 code
preg_replace_callback(
[
'~\$[a-z_][a-z\d_]*~i',
'~=~',
'~[\d]+~',
'~;~',
'~//.*~'
],
function ($match) use (&$tokenStream) {
if (strpos($match[0], '$') === 0) {
$tokenStream[] = ['T_VARIABLE', $match[0]];
} elseif (strpos($match[0], '=') === 0) {
$tokenStream[] = ['T_ASSIGN', $match[0]];
} elseif (ctype_digit($match[0])) {
$tokenStream[] = ['T_NUM', $match[0]];
} elseif (strpos($match[0], ';') === 0) {
$tokenStream[] = ['T_TERMINATE_STMT', $match[0]];
} elseif (strpos($match[0], '//') === 0) {
$tokenStream[] = ['T_COMMENT', $match[0]];
}
},
$input
);
// PHP 7+ code
preg_replace_callback_array(
[
'~\$[a-z_][a-z\d_]*~i' => function ($match) use (&$tokenStream) {
$tokenStream[] = ['T_VARIABLE', $match[0]];
},
'~=~' => function ($match) use (&$tokenStream) {
$tokenStream[] = ['T_ASSIGN', $match[0]];
},
'~[\d]+~' => function ($match) use (&$tokenStream) {
$tokenStream[] = ['T_NUM', $match[0]];
},
'~;~' => function ($match) use (&$tokenStream) {
$tokenStream[] = ['T_TERMINATE_STMT', $match[0]];
},
'~//.*~' => function ($match) use (&$tokenStream) {
$tokenStream[] = ['T_COMMENT', $match[0]];
}
],
$input
);
CSPRNG 系列函数
该特性涵盖两个函数,用于生成安全的整形与字符串,主要用于密码场景。它提供了简单的 API 和平台无关性。
string random_bytes(int length);
int random_int(int min, int max);
两个函数在没有足够的随机性时会报 E_WARNING 错误并且返回 false。
list() 可以打开对象实现 ArrayAccess
暂无