1. 程式人生 > >php7新特性的理解和比較

php7新特性的理解和比較

int 大於 spl dep csp sco 對象 vat empty

1. null合並運算符(??)

??語法: 如果變量存在且值不為NULL,它就會返回自身的值,否則返回它的第二個操作數.

 1 //php7以前  if判斷
 2 if(empty($_GET[‘param‘])) {
 3       $param = 1;
 4 }else{
 5     $param = $_GET[‘param‘];
 6 }
 7 
 8 //php7以前  三元運算符
 9 $param = empty($_GET[‘param‘]) ? 1 : $_GET[‘param‘];
10 
11 //PHP7  null合並運算符
12 $param
= $_GET[‘param‘] ?? 1;//1

2. define() 定義常量數組

 1 //php7以前
 2 define("CONTENT", "hello world");
 3 echo CONTENT;//hello world
 4 
 5 //PHP7
 6 define(‘ANIMALS‘, [
 7     ‘dog‘,
 8     ‘cat‘,
 9     ‘bird‘
10 ]);
11 echo ANIMALS[2];//bird
12 
13 //PHP7 類外也可使用const來定義常量
14 const CONSTANT = ‘Hello World‘; 
15 echo CONSTANT;//Hello World

3. 組合比較符(<=>)

組合比較符用於比較兩個表達式.當$a小於、等於或大於$b時它分別返回-1、0或1. 比較的原則是沿用PHP的常規比較規則進行的.

 1 //整數
 2 echo 1 <=> 1; // 0
 3 echo 1 <=> 2; // -1
 4 echo 2 <=> 1; // 1
 5 
 6 //浮點數
 7 echo 1.5 <=> 1.5; // 0
 8 echo 1.5 <=> 2.5; // -1
 9 echo 2.5 <=> 1.5; //
1 10 11 //字符串 12 echo "a" <=> "a"; // 0 13 echo "a" <=> "b"; // -1 14 echo "b" <=> "a"; // 1

4. 變量類型聲明

兩種模式: 強制(默認)和嚴格模式. 可以使用下列類型參數: string,int,float,bool

 1 //... 操作符: 表示這是一個可變參數. php5.6及以上的版本可使用: 函數定義的時候變量前使用.
 2 function intSum(int ...$ints){
 3     return array_sum($ints);
 4 }
 5 var_dump(intSum(2,‘3.5‘));//5
 6 
 7 //嚴格模式
 8 //模式聲明:declare(strict_types=1);  默認情況值為0,值為1代表為嚴格校驗的模式 
 9 declare(strict_types=1);
10 function add(int $a,int $b){
11     return $a+$b;
12 }
13 var_dump(add(2,‘3.5‘)); //Fatal error: Uncaught TypeError: Argument 2 passed to add() must be of the type integer

5. 返回值類型聲明

增加返回類型聲明的支持.類似於參數類型聲明.(用法在函數定義的後面加 :類型名)

1 //有效的返回類型
2 declare(strict_types = 1);
3 function getInt(int $value): int {
4   return $value;
5 }
6 print(getInt(6));//6

1 //無效返回類型
2 declare(strict_types = 1);
3 function getNoInt(int $value): int {
4   return $value+‘2.5‘;
5 }
6 print(getNoInt(6));//Fatal error: Uncaught TypeError: Return value of getNoInt() must be of the type integer

6. 匿名類

允許new class {} 創建一個匿名的對象.

 1 <?php
 2 //php7以前 接口實現
 3 interface User{
 4     public function getDiscount();
 5 }
 6 class VipUser implements User{
 7     //折扣系數
 8     private $discount = 0.6;
 9     public function getDiscount() {
10         return $this->discount;
11     }
12 }
13 class Goods{
14     private $price = 200;
15     private $objectVipUser;
16     //User接口VipUser類實現
17     public function getUserData($User){
18         $this->objectVipUser = $User;
19         $discount = $this->objectVipUser->getDiscount();
20         echo "商品價格:".$this->price*$discount;
21     }
22 }
23 $display = new Goods();
24 //常規實例化接口實現對象
25 $display ->getUserData(new VipUser);//商品價格:120

 1 <?php
 2 //php7 創建一個匿名的對象
 3 interface User{
 4     public function getDiscount();
 5 }
 6 class Goods{
 7     private $price = 200;
 8     private $objectVipUser;
 9     public function getUserData($User){
10         $this->objectVipUser = $User;
11         $discount = $this->objectVipUser->getDiscount();
12         echo "商品價格:".$this->price*$discount;
13     }
14 }
15 $display = new Goods();
16 //new匿名對象實現user接口
17 $display ->getUserData(new class implements User{
18     private $discount = 0.6;
19     public function getDiscount() {
20         return $this->discount;
21     }
22 });//商品價格:120

7. Closure::call()

Closure::call() 方法被添加為一個簡短的方式來臨時綁定一個對象作用域到一個閉包並調用它. 與PHP5的bindTo相比.它的性能要快得多.

 1 <?php
 2 //php7以前
 3 class A {
 4     private  $attribute = ‘hello world‘;
 5 }
 6 
 7 $getClosure = function(){
 8     return $this->attribute;
 9 };
10 
11 $getAttribute = $getClosure->bindTo(new A, ‘A‘);//中間層閉包
12 echo $getAttribute();//hello world

 1 <?php
 2 //PHP7
 3 class A {
 4     private  $attribute = ‘hello world‘;
 5 }
 6 
 7 $getClosure = function(){
 8     return $this->attribute;
 9 };
10 
11 echo $getClosure->call(new A);//hello world

8. unserialize()

unserialize()函數:過濾的特性,可以防止非法數據進行代碼註入,提供了更安全的反序列化數據

 1 <?php 
 2 class A{  
 3    public $name = ‘admin_a‘;    
 4 } 
 5 class B{ 
 6    public $name = ‘admin_b‘; 
 7 } 
 8 
 9 $objA = new A(); 
10 $objB = new B(); 
11 
12 $serializedObjA = serialize($objA); 
13 $serializedObjB = serialize($objB); 
14 
15 
16 //默認行為是接收所有類; 第二個參數可以忽略
17 $dataA = unserialize($serializedObjA , ["allowed_classes" => true]); 
18 var_dump($dataA);//object(A)#3 (1) { ["name"]=> string(7) "admin_a" }
19 
20 //如果allowed_classes設置為false,unserialize會將所有對象轉換為__PHP_Incomplete_Class對象 
21 $dataA = unserialize($serializedObjA , ["allowed_classes" => false]); 
22 var_dump($dataA);//object(__PHP_Incomplete_Class)#4 (2) { ["__PHP_Incomplete_Class_Name"]=> string(1) "A" ["name"]=> string(7) "admin_a" }
23 
24 //轉換所有對象到 __PHP_Incomplete_Class對象,除了對象"B"
25 $dataB = unserialize($serializedObjB , ["allowed_classes" => ["B"]]); 
26 var_dump($dataB);//object(B)#3 (1) { ["name"]=> string(7) "admin_b" }

9. IntlChar

IntlChar:提供了一些可用於訪問Unicode字符信息的實用方法的訪問. 註意:必須安裝Intl擴展才能使用!

1 var_dump(IntlChar::CODEPOINT_MAX);//int(1114111) 
2 echo ‘<br>‘;
3 var_dump(IntlChar::charName(‘+‘));//string(9) "PLUS SIGN" 
4 echo ‘<br>‘;
5 var_dump(IntlChar::ispunct(‘?‘));//bool(true)

10. CSPRNG

CSPRNG 函數提供一種簡單的機制來生成密碼學上強壯的隨機數.

random_bytes() -加密生存被保護的偽隨機字符串.

random_int() -加密生存被保護的偽隨機整數.

1 $bytes = random_bytes(8); 
2 echo(bin2hex($bytes));//隨機2073a110a2e3c497
3 echo ‘<br>‘;
4 echo(random_int(1, 999));//隨機786
5 echo ‘<br>‘;
6 print(random_int(-999, -1));//隨機-357 

11. use 語句

可以使用單個use語句從相同的命名空間導入類,函數和常量,而不是使用多個use語句.

 1 //PHP7之前
 2 use some\namespace\ClassA;
 3 use some\namespace\ClassB;
 4 use some\namespace\ClassC as C;
 5 use function some\namespace\fn_a;
 6 use function some\namespace\fn_b;
 7 use function some\namespace\fn_c;
 8 use const some\namespace\ConstA;
 9 use const some\namespace\ConstB;
10 use const some\namespace\ConstC;
11 
12 // PHP7之後
13 use some\namespace\{ClassA, ClassB, ClassC as C};
14 use function some\namespace\{fn_a, fn_b, fn_c};
15 use const some\namespace\{ConstA, ConstB, ConstC};

12. intdiv

新增加intdiv()函數,接收兩個參數,返回值為第一個參數除於第二個參數的值並取整.

1 echo intdiv(8,4);//2
2 echo intdiv(10,4);//2
3 echo intdiv(5,10);//0

13. PHP7 錯誤處理

PHP7 改變了大多數錯誤的報告方式.不同於PHP5的傳統錯誤報告機制,現在大多數錯誤被作為Error異常拋出.
這種Error異常可以像普通異常一樣被try / catch塊所捕獲. 如果沒有匹配的try / catch塊,則調用異常處理函數(由 set_exception_handler() 註冊)進行處理.
如果尚未註冊異常處理函數,則按照傳統方式處理:被報告為一個致命錯誤(Fatal Error).
Error類並不是從Exception類擴展出來的,所以用catch (Exception $e) { ... } 這樣的代碼是捕獲不到Error的.你可以用 catch (Error $e) { ... } 這樣的代碼,
或者通過註冊異常處理函數( set_exception_handler())來捕獲Error.
 1 <?php
 2 //php7以前 自定義異常處理
 3 class getException extends Exception{
 4     public function errorMsg(){
 5         return ‘錯誤的信息‘.$this->getMessage().‘<br>錯誤的代碼‘.$this->getCode();
 6     }
 7 }
 8 
 9 try {
10     $num =10;
11     if($num > 1) {
12         throw new getException($num,404);
13     }
14 } catch (getException $e) {
15     echo $e->errorMsg();
16 }

1 <?php  
2 //php7 異常處理
3 try {
4     test();
5 }catch(Error $e) {
6     echo $e->getMessage();//Call to undefined function test()
7 }

php7新特性的理解和比較