前言
本文是一篇講座聽後+後續研究的總結。
話說當年追時髦,php7一出就給電腦立馬裝上了,php5和php7共存,也是立馬寫了個超級耗時間的循環腳本測了一番,確實php7給力不少,而後也是注意了一些新增的特性與一些丟棄掉的用法。
因爲php升級乃頭等大事,公司近期纔打算升級,因此以前一直只能私下欣賞php7帶來的快感,負責升級的小夥伴搞了個分享,還挺全的,此處mark一下,看成筆記。php
主要研究問題:
1.PHP7帶來的好處
2.PHP7帶來的新東西
3.PHP7帶來的廢棄
4.PHP7帶來的變動
5.如何充分發揮PHP7的性能
6。如何更好的寫代碼來迎接PHP7?
7.如何升級當前項目代碼來兼容PHP7?html
PHP7帶來的好處
是的,性能上的大幅度提高,能夠省機器,能夠省錢。 mysql
PHP7帶來的新東西
1.類型的聲明。
可使用字符串(string), 整數 (int), 浮點數 (float), 以及布爾值 (bool),來聲明函數的參數類型與函數返回值。git
declare(strict_types=1);
function add(int $a, int $b): int {
return $a+$b;
}程序員
echo add(1, 2);
echo add(1.5, 2.6);
1
2
3
4
5
6
7
php5是沒法執行上面代碼的,php7執行的時候會先輸出一個3和一個報錯( Argument 1 passed to add() must be of the type integer, float given);github
標量類型聲明 有兩種模式: 強制 (默認) 和 嚴格模式。
declare(strict_types=1),必須放在文件的第一行執行代碼,當前文件有效!正則表達式
2.set_exception_handler() 再也不保證收到的必定是 Exception 對象
在 PHP 7 中,不少致命錯誤以及可恢復的致命錯誤,都被轉換爲異常來處理了。 這些異常繼承自 Error 類,此類實現了 Throwable 接口 (全部異常都實現了這個基礎接口)。sql
PHP7進一步方便開發者處理, 讓開發者對程序的掌控能力更強. 由於在默認狀況下, Error會直接致使程序中斷, 而PHP7則提供捕獲而且處理的能力, 讓程序繼續執行下去, 爲程序員提供更靈活的選擇。數組
3.新增操做符「<=>」
語法:$c = $a <=> $b安全
若是$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';
1
2
3
4
5
5.define() 定義常量數組
define('ARR',['a','b']);
echo ARR[1];// a
1
2
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
1
2
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};
1
2
3
4
5
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
1
2
3
4
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
1
2
3
4
5
6
7
8
9
10
11
12
3.參數處理機制修改
不支持重複參數命名
function func(a,a,b, c,c,c) {} ;hui報錯
func_get_arg()和func_get_args()這兩個方法返回參數當前的值, 而不是傳入時的值, 當前的值有可能會被修改
因此須要注意,在函數第一行最好就給記錄下來,不然後續有修改的話,再讀取就不是傳進來的初始值了。
function foo($x) {
$x++;
echo func_get_arg(0);
}
foo(1); //返回2
1
2
3
4
5
4.foreach修改
foreach()循環對數組內部指針再也不起做用
$arr = [1,2,3];
foreach ($arr as &$val) {
echo current($arr);// php7 全返回0
}
1
2
3
4
按照值進行循環的時候, foreach是對該數組的拷貝操做
$arr = [1,2,3];
foreach ($arr as $val) {
unset($arr[1]);
}
var_dump($arr);
1
2
3
4
5
最新的php7依舊會打印出[1,2,3]。(ps:7.0.0不行)
老的會打印出[1,3]
按照引用進行循環的時候, 對數組的修改會影響循環
$arr = [1];
foreach ($arr as $val) {
var_dump($val);
$arr[1]=2;
}
1
2
3
4
5
最新的php7依舊會追加新增元素的循環。(ps:7.0.0不行)
5. list修改
再也不按照相反的順序賦值
//$arr將會是[1,2,3]而不是以前的[3,2,1]
list($arr[], $arr[], $arr[]) = [1,2,3];
1
2
再也不支持字符串拆分功能
// $x = null 而且 $y = null
$str = 'xy';
list($x, $y) = $str;
1
2
3
空的list()賦值再也不容許
list() = [123];
1
list()如今也適用於數組對象
list($a, $b) = (object)new ArrayObject([0, 1]);
1
6.變量處理機制修改
對變量、屬性和方法的間接調用如今將嚴格遵循從左到右的順序來解析,而不是以前的混雜着幾個特殊案例的狀況。 下面這張表說明了這個解析順序的變化。
引用賦值時自動建立的數組元素或者對象屬性順序和之前不一樣了
$arr = [];
$arr['a'] = &$arr['b'];
$arr['b'] = 1;
// php7: ['a' => 1, 'b' => 1]
// php5: ['b' => 1, 'a' => 1]
1
2
3
4
5
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?
不使用php7廢棄的方法,擴展
使用2個版本都兼容的語法特性【 list ,foreach, func_get_arg 等】
如何升級當前項目代碼來兼容PHP7?
逐步剔除php7不支持的代碼
檢測工具:https://github.com/sstalle/php7cc
更多資料
官方5.6.x移植7.0.x 文檔
Laruence 讓PHP7達到最高性能的tips
博客-PHP7特徵