一篇寫給準備升級PHP7的小夥伴的文章

PHP7帶來的好處

是的,效能上的大幅度提升,可以省機器,可以省錢。

一篇寫給準備升級PHP7的小夥伴的文章

PHP7帶來的新東西

1。型別的宣告。

可以使用字串(string), 整數 (int), 浮點數 (float), 以及布林值 (bool),來宣告函式的引數型別與函式返回值。

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);

標量型別宣告 有兩種模式: 強制 (預設) 和 嚴格模式。

declare(strict_types=1),必須放在檔案的第一行執行程式碼,當前檔案有效!

2。set_exception_handler() 不再保證收到的一定是 Exception 物件

在 PHP 7 中,很多致命錯誤以及可恢復的致命錯誤,都被轉換為異常來處理了。 這些異常繼承自 Error 類,此類實現了 Throwable 介面 (所有異常都實現了這個基礎介面)。

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’;

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,

a,b, c,

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。變數處理機制修改

對變數、屬性和方法的間接呼叫現在將嚴格遵循從左到右的順序來解析,而不是之前的混雜著幾個特殊案例的情況。 下面這張表說明了這個解析順序的變化。

一篇寫給準備升級PHP7的小夥伴的文章

引用賦值時自動建立的陣列元素或者物件屬性順序和以前不同了

$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 (根據系統記憶體決定)

圖片描述(最多50字)

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

一篇寫給準備升級PHP7的小夥伴的文章