PHP版本-對應的新特性
最近做的專案使用了 php7,但感覺有很多新特性沒有用起來。就想總結一下,一些可能會用到的新特性。之前使用的環境是 php5.3,所有也會有 php5.5 和 php5.6 的特性總結進來,這裡只列出我覺得在專案中可能用到的特性。
1.Generators (PHP 5 >= 5.5.0, PHP 7)
通過新增 yield 關鍵字支援了 generators,Generators 提供了一個更簡單的方法實現迭代器,不需要實現 Iterator 介面。
function xrange($start, $limit, $step = 1) {
for ($i = $start ; $i <= $limit; $i += $step) {
yield $i;
}
}
echo 'Single digit odd numbers: ';
/* 注意儲存在記憶體中的陣列絕不會被建立或返回 */
foreach (xrange(1, 9, 2) as $number) {
echo "$number ";
}
// 執行結果
Single digit odd numbers: 1 3 5 7 9
2.新增 finally 關鍵字 (PHP 5 >= 5.5.0, PHP 7)
try-catch 現在支援 finally
3.foreach 現在支援 list() (PHP 5 >= 5.5.0, PHP 7)
$array = [
[1, 2],
[3, 4],
];
foreach ($array as list($a, $b)) {
echo "A: $a; B: $b\n";
}
// 輸出結果
A: 1; B: 2
A: 3; B: 4
4.array_column (PHP 5 >= 5.5.0, PHP 7)
array array_column ( array $input , mixed $column_key [, mixed $index_key ] )
array_column()返回input陣列中鍵值為column_key的列, 如果指定了可選引數index_key,那麼input陣列中的這一列的值將作為返回陣列中對應值的鍵。
$records = array(
array(
'id' => 2135,
'first_name' => 'John',
'last_name' => 'Doe',
),
array(
'id' => 3245,
'first_name' => 'Sally',
'last_name' => 'Smith',
),
array(
'id' => 5342,
'first_name' => 'Jane',
'last_name' => 'Jones',
),
array(
'id' => 5623,
'first_name' => 'Peter',
'last_name' => 'Doe',
)
);
$first_names = array_column($records, 'first_name');
print_r($first_names);
// 輸出結果
Array
(
[0] => John
[1] => Sally
[2] => Jane
[3] => Peter
)
$last_names = array_column($records, 'last_name', 'id');
print_r($last_names);
// 輸出結果
Array
(
[2135] => Doe
[3245] => Smith
[5342] => Jones
[5623] => Doe
)
5.使用表示式定義常量 (PHP 5 >= 5.6.0, PHP 7)*
在之前的 PHP 版本中, 必須使用靜態值來定義常量,宣告屬性以及指定函式引數預設值。 現在你可以使用包括數值、字串字面量以及其他常量在內的數值表示式來 定義常量、宣告屬性以及設定函式引數預設值。
const ONE = 1;
const TWO = ONE * 2;
class C {
const THREE = TWO + 1;
const ONE_THIRD = ONE / self::THREE;
const SENTENCE = 'The value of THREE is '.self::THREE;
public function f($a = ONE + self::THREE) {
return $a;
}
}
echo (new C)->f()."\n";
echo C::SENTENCE;
現在可以通過 const 關鍵字來定義型別為 array 的常量。
const ARR = ['a', 'b'];
echo ARR[0];
6.使用 … 運算子定義變長引數函式 (PHP 5 >= 5.6.0, PHP 7)
現在可以不依賴 func_get_args(), 使用 … 運算子 來實現 變長引數函式。
function f($req, $opt = null, ...$params) {
// $params 是一個包含了剩餘引數的陣列
printf('$req: %d; $opt: %d; number of params: %d'."\n",
$req, $opt, count($params));
}
f(1);
f(1, 2);
f(1, 2, 3);
f(1, 2, 3, 4);
f(1, 2, 3, 4, 5);
7.使用 … 運算子進行引數展開 (PHP 5 >= 5.6.0, PHP 7)
在呼叫函式的時候,使用 … 運算子, 將 陣列 和 可遍歷 物件展開為函式引數。 在其他程式語言,比如 Ruby中,這被稱為連線運算子
function add($a, $b, $c) {
return $a + $b + $c;
}
$operators = [2, 3];
echo add(1, ...$operators);
// 輸出6
8.use function 以及 use const (PHP 5 >= 5.6.0, PHP 7)
use 運算子 被進行了擴充套件以支援在類中匯入外部的函式和常量。 對應的結構為 use function 和 use const。
namespace Name\Space {
const FOO = 42;
function f() { echo __FUNCTION__."\n"; }
}
namespace {
use const Name\Space\FOO;
use function Name\Space\f;
echo FOO."\n";
f();
}
// 輸出結果
42
Name\Space\f
9.__debugInfo() (PHP 5 >= 5.6.0, PHP 7)
加入 __debugInfo(), 當使用 var_dump() 輸出物件的時候, 可以用來控制要輸出的屬性和值。
class C {
private $prop;
public function __construct($val) {
$this->prop = $val;
}
public function __debugInfo() {
return [
'propSquared' => $this->prop ** 2,
];
}
}
var_dump(new C(42));
// 輸出結果
object(C)#1 (1) {
["propSquared"]=>
int(1764)
}
10.標量型別宣告 (PHP 7)
標量型別宣告 有兩種模式: 強制 (預設) 和 嚴格模式。 現在可以使用下列型別引數(無論用強制模式還是嚴格模式): 字串(string), 整數 (int), 浮點數 (float), 以及布林值 (bool)。它們擴充了PHP5中引入的其他型別:類名,介面,陣列和 回撥型別。
// Coercive mode
function sumOfInts(int ...$ints)
{
return array_sum($ints);
}
var_dump(sumOfInts(2, '3', 4.1));
// 輸出結果
int(9)
11.返回值型別宣告 (PHP 7)
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]));
// 輸出結果
Array
(
[0] => 6
[1] => 15
[2] => 24
)
12.null合併運算子 (PHP 7)
由於日常使用中存在大量同時使用三元表示式和 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';
13.太空船操作符(組合比較符)(PHP 7)
太空船操作符用於比較兩個表示式。當
// 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
14.通過 define() 定義常量陣列 (PHP 7)
define('ANIMALS', [
'dog',
'cat',
'bird'
]);
echo ANIMALS[1]; // outputs "cat"
15.匿名類 (PHP 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());
16.Closure::call() (PHP 7)
Closure::call() 現在有著更好的效能,簡短幹練的暫時繫結一個方法到物件上閉包並呼叫它。
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);
17.Group use declarations (PHP 7)
從同一 namespace 匯入的類、函式和常量現在可以通過單個 use 語句 一次性匯入了。
// 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};
18.為unserialize()提供過濾 (PHP 7)
這個特性旨在提供更安全的方式解包不可靠的資料。它通過白名單的方式來防止潛在的程式碼注入。
// 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]);