您好,登錄后才能下訂單哦!
本文小編為大家詳細介紹“PHP7.X版本的新特性有哪些”,內容詳細,步驟清晰,細節處理妥當,希望這篇“PHP7.X版本的新特性有哪些”文章能幫助大家解決疑惑,下面跟著小編的思路慢慢深入,一起來學習新知識吧。
標量類型聲明有兩種模式: 強制 (默認) 和 嚴格模式。 現在可以使用下列類型參數(無論用強制模式還是嚴格模式): 字符串(string), 整數 (int), 浮點數 (float), 以及布爾值 (bool)。它們擴充了PHP5中引入的其他類型:類名,接口,數組和 回調類型。
PHP標量包含: 字符串(string
), 整數 (int
), 浮點數 (float
), 以及布爾值 (bool
)。
例如下面我們定義一個形式參數為整數的參數。(正確的如下)
<?php //Enter your code here, enjoy! function sumOfInts(int ...$ints) { return array_sum($ints); } var_dump(sumOfInts(2, '3', 4.1));
輸出:
int(9)
例如下面我們定義一個形式參數為整數的參數。(錯誤的如下)
<?php //Enter your code here, enjoy! function sumOfInts(int ...$ints) { return array_sum($ints); } var_dump(sumOfInts(2, 'error', 4.1));//參數里面有字符串,我們聲明的是整數
輸出錯誤信息:
<br /> <b>Fatal error</b>: Uncaught TypeError: Argument 2 passed to sumOfInts() must be of the type integer, string given, called in [...][...] on line 8 and defined in [...][...]:3 Stack trace: #0 [...][...](8): sumOfInts(2, 'error', 4.1) #1 {main} thrown in <b>[...][...]</b> on line <b>3</b><br />
PHP 7 增加了對返回類型聲明的支持。 類似于參數類型聲明,返回類型聲明指明了函數返回值的類型。可用的類型與參數聲明中可用的類型相同。
例如下面我們定義一個返回值為數組的函數。
<?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()的情況, 我們添加了null合并運算符 (??) 這個語法糖。如果變量存在且值不為NULL, 它就會返回自身的值,否則返回它的第二個操作數。
<?php // 如果$_GET['user']不存在就執行nobody賦值給$username $username = $_GET['user'] ?? 'nobody'; // 上面的語句相當于下面的語句 $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'; ?>
太空船操作符用于比較兩個表達式。當$a小于、等于或大于$b時它分別返回-1、0或1。 比較的原則是沿用 PHP 的常規比較規則進行的。
<?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 ?>
Array 類型的常量現在可以通過 define() 來定義。在 PHP5.6 中僅能通過 const 定義。
<?php define('ANIMALS', [ 'dog', 'cat', 'bird' ]); echo ANIMALS[1]; // 輸出 "cat" ?>
現在支持通過new class 來實例化一個匿名類,這可以用來替代一些“用后即焚”的完整類定義。
<?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) { }
這接受一個以16進制形式的 Unicode codepoint,并打印出一個雙引號或heredoc包圍的 UTF-8 編碼格式的字符串。 可以接受任何有效的 codepoint,并且開頭的 0 是可以省略的。
echo "\u{aa}"; echo "\u{0000aa}"; echo "\u{9999}";
以上例程會輸出:
a a (same as before but with optional leading 0's) 香
Closure::call() 現在有著更好的性能,簡短干練的暫時綁定一個方法到對象上閉包并調用它。
<?php 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);
以上例程會輸出:
1 1
這個特性旨在提供更安全的方式解包不可靠的數據。它通過白名單的方式來防止潛在的代碼注入。
// 將所有的對象都轉換為 __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]);
新增加的 IntlChar 類旨在暴露出更多的 ICU 功能。這個類自身定義了許多靜態方法用于操作多字符集的 unicode 字符。
<?php printf('%x', IntlChar::CODEPOINT_MAX); echo IntlChar::charName('@'); var_dump(IntlChar::ispunct('!'));
以上例程會輸出:
10ffff COMMERCIAL AT bool(true)
若要使用此類,請先安裝Intl擴展
預期是向后兼用并增強之前的 assert() 的方法。 它使得在生產環境中啟用斷言為零成本,并且提供當斷言失敗時拋出特定異常的能力。
老版本的API出于兼容目的將繼續被維護,assert()現在是一個語言結構,它允許第一個參數是一個表達式,而不僅僅是一個待計算的 string或一個待測試的boolean。
<?php ini_set('assert.exception', 1); class CustomError extends AssertionError {} assert(false, new CustomError('Some error message')); ?>
以上例程會輸出:
Fatal error: Uncaught CustomError: Some error message
關于這個特性的完整說明,包括如何在開發和生產環境中配置它,可以在assert()的 expectations section章節找到。
從同一 namespace 導入的類、函數和常量現在可以通過單個 use 語句 一次性導入了。
<?php // PHP 7 之前的代碼 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+ 及更高版本的代碼 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 對象或者 array。
<?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() 可以接受一個 array 作為參數, 用來覆蓋 php.ini 文件中設置的 會話配置選項。
在調用 session_start() 的時候, 傳入的選項參數中也支持 session.lazy_write 行為, 默認情況下這個配置項是打開的。它的作用是控制 PHP 只有在會話中的數據發生變化的時候才 寫入會話存儲文件,如果會話中的數據沒有發生改變,那么 PHP 會在讀取完會話數據之后, 立即關閉會話存儲文件,不做任何修改,可以通過設置 read_and_close 來實現。
例如,下列代碼設置 session.cache_limiter 為 private,并且在讀取完畢會話數據之后馬上關閉會話存儲文件。
<?php session_start([ 'cache_limiter' => 'private', 'read_and_close' => true, ]); ?>
在 PHP 7 之前,當使用 preg_replace_callback() 函數的時候, 由于針對每個正則表達式都要執行回調函數,可能導致過多的分支代碼。 而使用新加的 preg_replace_callback_array() 函數, 可以使得代碼更加簡潔。
現在,可以使用一個關聯數組來對每個正則表達式注冊回調函數, 正則表達式本身作為關聯數組的鍵, 而對應的回調函數就是關聯數組的值。
新加入兩個跨平臺的函數: random_bytes() 和 random_int() 用來產生高安全級別的隨機字符串和隨機整數。
可以使用 list() 函數來展開實現了 ArrayAccess 接口的對象 ?
在之前版本中,list() 函數不能保證 正確的展開實現了 ArrayAccess 接口的對象, 現在這個問題已經被修復。
允許在克隆表達式上訪問對象成員,例如: (clone $foo)->bar()。
參數以及返回值的類型現在可以通過在類型前加上一個問號使之允許為空。 當啟用這個特性時,傳入的參數或者函數返回的結果要么是給定的類型,要么是 null 。
<?php function testReturn(): ?string { return 'elePHPant'; } var_dump(testReturn()); function testReturn(): ?string { return null; } var_dump(testReturn()); function test(?string $name) { var_dump($name); } test('elePHPant'); test(null); test();
以上例程會輸出:
string(10) "elePHPant" NULL string(10) "elePHPant" NULL Uncaught Error: Too few arguments to function test(), 0 passed in...
一個新的返回值類型void被引入。 返回值聲明為 void 類型的方法要么干脆省去 return 語句,要么使用一個空的 return 語句。 對于 void 函數來說,NULL
不是一個合法的返回值。
<?php function swap(&$left, &$right) : void { if ($left === $right) { return; } $tmp = $left; $left = $right; $right = $tmp; } $a = 1; $b = 2; var_dump(swap($a, $b), $a, $b);
以上例程會輸出:
null int(2) int(1)
試圖去獲取一個 void 方法的返回值會得到 NULL ,并且不會產生任何警告。這么做的原因是不想影響更高層次的方法。
<?php class Sky8g { const PUBLIC_CONST_A = 1; public const PUBLIC_CONST_B = 2; protected const PROTECTED_CONST = 3; private const PRIVATE_CONST = 4; }
現在引入了一個新的被稱為iterable的偽類 (與callable類似)。 這可以被用在參數或者返回值類型中,它代表接受數組或者實現了Traversable接口的對象。 至于子類,當用作參數時,子類可以收緊父類的iterable類型到array 或一個實現了Traversable的對象。對于返回值,子類可以拓寬父類的 array或對象返回值類型到iterable。
<?php function iterator(iterable $iter) { foreach ($iter as $val) { // } }
一個catch語句塊現在可以通過管道字符(|)來實現多個異常的捕獲。 這對于需要同時處理來自不同類的不同異常時很有用。
<?php try { // some code } catch (FirstException | SecondException $e) { // handle first and second exceptions }
現在list()和它的新的[]語法支持在它內部去指定鍵名。這意味著它可以將任意類型的數組 都賦值給一些變量(與短數組語法類似)
<?php $data = [ ["id" => 1, "name" => 'Tom'], ["id" => 2, "name" => 'Fred'], ]; // list() style list("id" => $id1, "name" => $name1) = $data[0]; // [] style ["id" => $id1, "name" => $name1] = $data[0]; // list() style foreach ($data as list("id" => $id, "name" => $name)) { // logic here with $id and $name } // [] style foreach ($data as ["id" => $id, "name" => $name]) { // logic here with $id and $name }
現在所有支持偏移量的字符串操作函數 都支持接受負數作為偏移量,包括通過[]或{}操作字符串下標。在這種情況下,一個負數的偏移量會被理解為一個從字符串結尾開始的偏移量。
<?php var_dump("abcdef"[-2]); var_dump(strpos("aabbcc", "b", -3));
以上例程會輸出:
string (1) "e" int(3)
這種新的對象類型object
, 引進了可用于逆變(contravariant)參數輸入和協變(covariant)返回任何對象類型。
<?php function test(object $obj) : object { return new SplQueue(); } test(new StdClass());
擴展文件不再需要通過文件加載 (Unix下以.so
為文件擴展名,在Windows下以
.dll
為文件擴展名) 進行指定。可以在php.ini
配置文件進行啟用, 也可以使用 dl()
函數進行啟用。
當一個抽象類繼承于另外一個抽象類的時候,繼承后的抽象類可以重寫被繼承的抽象類的抽象方法。
<?php abstract class A { abstract function test(string $s); } abstract class B extends A { // overridden - still maintaining contravariance for parameters and covariance for return abstract function test($s) : int; }
Argon2 已經被加入到密碼散列(password hashing) API (這些函數以 password_ 開頭), 以下是暴露出來的常量:
PASSWORD_ARGON2I
PASSWORD_ARGON2_DEFAULT_MEMORY_COST
PASSWORD_ARGON2_DEFAULT_TIME_COST
PASSWORD_ARGON2_DEFAULT_THREADS
當你準備支持多語言字符集,PDO的字符串類型已經擴展支持國際化的字符集。以下是擴展的常量:
PDO::PARAM_STR_NATL
PDO::PARAM_STR_CHAR
PDO::ATTR_DEFAULT_STR_PARAM
這些常量通過PDO::PARAM_STR
利用位運算OR進行計算:
<?php $db->quote('über', PDO::PARAM_STR | PDO::PARAM_STR_NATL);
PDOStatement::debugDumpParams()
方法已經更新,當發送SQL到數據庫的時候,在一致性、行查詢(包括替換綁定占位符)將會顯示調試信息。這一特性已經加入到模擬調試中(在模擬調試打開時可用)。
LDAP 擴展已經新增了EXOP支持. 擴展暴露以下函數和常量:
ldap_parse_exop()
ldap_exop()
ldap_exop_passwd()
ldap_exop_whoami()
LDAP_EXOP_START_TLS
LDAP_EXOP_MODIFY_PASSWD
LDAP_EXOP_REFRESH
LDAP_EXOP_WHO_AM_I
LDAP_EXOP_TURN
sockets擴展現在具有查找地址信息的能力,且可以連接到這個地址,或者進行綁定和解析。為此添加了以下一些函數:
socket_addrinfo_lookup()
socket_addrinfo_connect()
socket_addrinfo_bind()
socket_addrinfo_explain()
重寫方法和接口實現的參數類型現在可以省略了。不過這仍然是符合LSP,因為現在這種參數類型是逆變的。
<?php interface A { public function Test(array $input); } class B implements A { public function Test($input){} // type omitted for $input }
命名空間可以在PHP 7中使用尾隨逗號進行分組引入。
<?php use Foo\Bar\{ Foo, Bar, Baz, };
多字節字符串數據表已更新為Unicode 11。
多字節字符串函數現在正確支持大于2GB的字符串。
多字節字符串擴展的性能得到了全面的顯著改善。最大的改進是大小寫轉換功能。
mb_ereg_*
函數現在支持命名捕捉。像mb_ereg()
這樣的匹配函數現在將使用它們的組號和名稱返回指定的捕獲,類似于PCRE:
<?php mb_ereg('(?<word>\w+)', '國', $matches); // => [0 => "國", 1 => "國", "word" => "國"]; ?>
另外,mb_ereg_replace()
現在支持\k<>和\k "
符號來引用替換字符串中的指定捕獲:
<?php mb_ereg_replace('\s*(?<word>\w+)\s*', "_\k<word>_\k'word'_", ' foo '); // => "_foo_foo_" ?>
\k<>和\k "
也可用于編號引用,也可用于大于9的組號。
類屬性現在支持類型聲明。
<?php class User { public int $id; public string $name; } ?>
上面的示例將強制執行$user->id
只能賦給整數值,而$user->name
只能賦給字符串值。
箭頭函數為使用隱式按值范圍綁定定義函數提供了一種簡寫語法。
<?php $factor = 10; $nums = array_map(fn($n) => $n * $factor, [1, 2, 3, 4]); // $nums = array(10, 20, 30, 40); ?>
<?php class A {} class B extends A {} class Producer { public function method(): A {} } class ChildProducer extends Producer { public function method(): B {} } ?>
只有在使用自動加載時,才可以使用全方差支持。在單個文件中,只有非循環類型引用是可能的,因為所有類在被引用之前都必須是可用的。
<?php $array['key'] ??= computeDefault(); // is roughly equivalent to if (!isset($array['key'])) { $array['key'] = computeDefault(); } ?>
<?php $parts = ['apple', 'pear']; $fruits = ['banana', 'orange', ...$parts, 'watermelon']; // ['banana', 'orange', 'apple', 'pear', 'watermelon']; ?>
<?php 6.674_083e-11; // float 299_792_458; // decimal 0xCAFE_F00D; // hexadecimal 0b0101_1111; // binary ?>
弱引用允許程序員保留對對象的引用,而不阻止對象被銷毀。
讀到這里,這篇“PHP7.X版本的新特性有哪些”文章已經介紹完畢,想要掌握這篇文章的知識點還需要大家自己動手實踐使用過才能領會,如果想了解更多相關內容的文章,歡迎關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。