Skip to content

Latest commit

 

History

History
398 lines (342 loc) · 11.9 KB

New-features.md

File metadata and controls

398 lines (342 loc) · 11.9 KB

PHP5.6.x 版本迁移至 7.0.x 版本

新的特性

标量类型声明

标量类型声明有两种模式:强制(默认)模式、严格模式。下列参数类型可以使用(无论用强制模式还是严格模式):字符串(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
)

详细的返回值声明相关的文档和示例代码可以查阅返回值声明文档。

NULL合并算子(操作符“??”)

空合并算子的操作符为 ?? ,已经作为一种语法糖用于日常需求中用于三元表达式,它与 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
?>

通过 define() 定义常量数组

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 转译语法

通过十六进制内容与双引号组成的字符串生成 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

为 unserialize() 提供过滤

这个特性意在提供更安全的方式解包不可靠的数据。通过白名单的方式来防止代码注入。

<?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() 函数。

使用 use 集体声明

在 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() 做整除

intdiv() 函数来处理整除,并返回一个整数。

<?php

var_dump(intdiv(10, 3));

上述代码输出:

int(3)

session_start() 选项

该特性给 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_array() 函数

这个新功能,当你使用 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

用户贡献记录

暂无