php7與以前的區別和更新【轉載提煉】

主要研究問題: php

1.PHP7帶來的好處 
2.PHP7帶來的新東西 
3.PHP7帶來的廢棄 
4.PHP7帶來的變動 
5.如何充分發揮PHP7的性能 
6.如何更好的寫代碼來迎接PHP7?
7.如何升級當前項目代碼來兼容PHP7css

PHP7帶來的好處

是的,性能上的大幅度提高,能夠省機器,能夠省錢。 html

 

PHP7帶來的新東西

1.類型的聲明。

可使用字符串(string), 整數 (int), 浮點數 (float), 以及布爾值 (bool),來聲明函數的參數類型與函數返回值。mysql

declare(strict_types=1);
function add(int $a, int $b): int {
    return $a+$b;
}

echo add(1, 2);
echo add(1.5, 2.6);

 

php5是沒法執行上面代碼的,php7執行的時候會先輸出一個3和一個報錯( Argument 1 passed to add() must be of the type integer, float given);git

標量類型聲明 有兩種模式: 強制 (默認) 和 嚴格模式。 
declare(strict_types=1),必須放在文件的第一行執行代碼,當前文件有效!程序員

2.set_exception_handler() 再也不保證收到的必定是 Exception 對象

在 PHP 7 中,不少致命錯誤以及可恢復的致命錯誤,都被轉換爲異常來處理了。 這些異常繼承自 Error 類,此類實現了 Throwable 接口 (全部異常都實現了這個基礎接口)。github

PHP7進一步方便開發者處理, 讓開發者對程序的掌控能力更強. 由於在默認狀況下, Error會直接致使程序中斷, 而PHP7則提供捕獲而且處理的能力, 讓程序繼續執行下去, 爲程序員提供更靈活的選擇。正則表達式

3.新增操做符「<=>」

語法:$c = $a <=> $bsql

若是$a > $b, $c 的值爲1數組

若是$a == $b, $c 的值爲0

若是$a < $b, $c 的值爲-1

4.新增操做符「??」

若是變量存在且值不爲NULL, 它就會返回自身的值,不然返回它的第二個操做數。

//原寫法 $username = isset($_GET['user]) ? $_GET['user] : 'nobody'; //如今 $username = $_GET['user'] ?? 'nobody';

5.define() 定義常量數組

define('ARR',['a','b']); echo ARR[1];// a

6.AST: Abstract Syntax Tree, 抽象語法樹

AST在PHP編譯過程做爲一箇中間件的角色, 替換原來直接從解釋器吐出opcode的方式, 讓解釋器(parser)和編譯器(compliler)解耦, 能夠減小一些Hack代碼, 同時, 讓實現更容易理解和可維護.

PHP5 : PHP代碼 -> Parser語法解析 -> OPCODE -> 執行 
PHP7 : PHP代碼 -> Parser語法解析 -> AST -> OPCODE -> 執行

參考: https://wiki.php.net/rfc/abstract_syntax_tree

7.匿名函數

$anonymous_func = function(){return 'function';}; echo $anonymous_func(); // 輸出function

8.Unicode字符格式支持(echo 「\u{9999}」)

9.Unserialize 提供過濾特性

防止非法數據進行代碼注入,提供了更安全的反序列化數據。

10.命名空間引用優化

// PHP7之前語法的寫法 
use FooLibrary\Bar\Baz\ClassA; use FooLibrary\Bar\Baz\ClassB; // PHP7新語法寫法 use FooLibrary\Bar\Baz\{ ClassA, ClassB};

 

PHP7帶來的廢棄

1.廢棄擴展

Ereg 正則表達式 
mssql 
mysql 
sybase_ct

2.廢棄的特性

不能使用同名的構造函數 
實例方法不能用靜態方法的方式調用

3.廢棄的函數

方法調用 
call_user_method() 
call_user_method_array()

應該採用call_user_func() 和 call_user_func_array()

加密相關函數

mcrypt_generic_end() 
mcrypt_ecb() 
mcrypt_cbc() 
mcrypt_cfb() 
mcrypt_ofb()

注意: PHP7.1 之後mcrypt_*序列函數都將被移除。推薦使用:openssl 序列函數

雜項

set_magic_quotes_runtime 
set_socket_blocking 
Split 
imagepsbbox() 
imagepsencodefont() 
imagepsextendfont() 
imagepsfreefont() 
imagepsloadfont() 
imagepsslantfont() 
imagepstext()

4.廢棄的用法

$HTTP_RAW_POST_DATA 變量被移除, 使用php://input來代

ini文件裏面再也不支持#開頭的註釋, 使用」;」

移除了ASP格式的支持和腳本語法的支持: <% 和 < script language=php >

PHP7帶來的變動

1.字符串處理機制修改

含有十六進制字符的字符串再也不視爲數字, 也再也不區別對待.

var_dump("0x123" == "291"); // false var_dump(is_numeric("0x123")); // false var_dump("0xe" + "0x1"); // 0 var_dump(substr("f00", "0x1")) // foo

2.整型處理機制修改

Int64支持, 統一不一樣平臺下的整型長度, 字符串和文件上傳都支持大於2GB. 64位PHP7字符串長度能夠超過2^31次方字節.

// 無效的八進制數字(包含大於7的數字)會報編譯錯誤 $i = 0681; // 老版本php會把無效數字忽略。 // 位移負的位置會產生異常 var_dump(1 >> -1); // 左位移超出位數則返回0 var_dump(1 << 64);// 0 // 右位移超出會返回0或者-1 var_dump(100 >> 32);// 0 var_dump(-100 >> 32);// -1 

3.參數處理機制修改

不支持重複參數命名

function func(a,b, c,c) {} ;hui報錯

func_get_arg()和func_get_args()這兩個方法返回參數當前的值, 而不是傳入時的值, 當前的值有可能會被修改

因此須要注意,在函數第一行最好就給記錄下來,不然後續有修改的話,再讀取就不是傳進來的初始值了。

function foo($x) { $x++; echo func_get_arg(0); } foo(1); //返回2

 

4.foreach修改

foreach()循環對數組內部指針再也不起做用

$arr = [1,2,3]; foreach ($arr as &$val) { echo current($arr);// php7 全返回0 }

按照值進行循環的時候, foreach是對該數組的拷貝操做

$arr = [1,2,3]; foreach ($arr as $val) { unset($arr[1]); } var_dump($arr);

最新的php7依舊會打印出[1,2,3]。(ps:7.0.0不行) 
老的會打印出[1,3]

按照引用進行循環的時候, 對數組的修改會影響循環

$arr = [1]; foreach ($arr as $val) { var_dump($val); $arr[1]=2; }

最新的php7依舊會追加新增元素的循環。(ps:7.0.0不行)

5. list修改

再也不按照相反的順序賦值

//$arr將會是[1,2,3]而不是以前的[3,2,1] list($arr[], $arr[], $arr[]) = [1,2,3];

再也不支持字符串拆分功能

// $x = null 而且 $y = null $str = 'xy'; list($x, $y) = $str;

空的list()賦值再也不容許

list() = [123];

list()如今也適用於數組對象

list($a, $b) = (object)new ArrayObject([0, 1]);

6.變量處理機制修改

對變量、屬性和方法的間接調用如今將嚴格遵循從左到右的順序來解析,而不是以前的混雜着幾個特殊案例的狀況。 下面這張表說明了這個解析順序的變化。

 

引用賦值時自動建立的數組元素或者對象屬性順序和之前不一樣了

$arr = []; $arr['a'] = &$arr['b']; $arr['b'] = 1; // php7: ['a' => 1, 'b' => 1] // php5: ['b' => 1, 'a' => 1]

7.雜項

1.debug_zval_dump() 如今打印 「int」 替代 「long」, 打印 「float」 替代 「double」

2.dirname() 增長了可選的第二個參數, depth, 獲取當前目錄向上 depth 級父目錄的名稱。

3.getrusage() 如今支持 Windows.mktime() and gmmktime() 函數再也不接受 is_dst 參數。

4.preg_replace() 函數再也不支持 「\e」 (PREG_REPLACE_EVAL). 應當使用 preg_replace_callback() 替代。

5.setlocale() 函數再也不接受 category 傳入字符串。 應當使用 LC_* 常量。

6.exec(), system() and passthru() 函數對 NULL 增長了保護.

7.shmop_open() 如今返回一個資源而非一個int, 這個資源能夠傳給shmop_size(), shmop_write(), shmop_read(), shmop_close() 和 shmop_delete().

8.爲了不內存泄露,xml_set_object() 如今在執行結束時須要手動清除 $parse。

9.curl_setopt 設置項CURLOPT_SAFE_UPLOAD變動

TRUE 禁用 @ 前綴在 CURLOPT_POSTFIELDS 中發送文件。 意味着 @ 能夠在字段中安全得使用了。 可以使用 CURLFile做爲上傳的代替。 
PHP 5.5.0 中添加,默認值 FALSE。 PHP 5.6.0 改默認值爲 TRUE。. PHP 7 刪除了此選項, 必須使用 CURLFile interface 來上傳文件。

如何充分發揮PHP7的性能

1.開啓Opcache

zend_extension=opcache.so 
opcache.enable=1 
opcache.enable_cli=1

2.使用GCC 4.8以上進行編譯

只有GCC 4.8以上PHP纔會開啓Global Register for opline and execute_data支持, 這個會帶來5%左右的性能提高(Wordpres的QPS角度衡量)

3.開啓HugePage (根據系統內存決定)

 

4.PGO (Profile Guided Optimization)

第一次編譯成功後,用項目代碼去訓練PHP,會產生一些profile信息,最後根據這些信息第二次gcc編譯PHP就能夠獲得量身定作的PHP7

須要選擇在你要優化的場景中: 訪問量最大的, 耗時最多的, 資源消耗最重的一個頁面.

參考: http://www.laruence.com/2015/06/19/3063.html 
參考: http://www.laruence.com/2015/12/04/3086.html

如何更好的寫代碼來迎接PHP7?

  1. 不使用php7廢棄的方法,擴展
  2. 使用2個版本都兼容的語法特性【 list ,foreach, func_get_arg 等】

如何升級當前項目代碼來兼容PHP7?

逐步剔除php7不支持的代碼

檢測工具:https://github.com/sstalle/php7cc

相關文章
相關標籤/搜索