PHP7
2015.12.3 發生了兩件大事, PHP7 問世了, Swift 開源了。
最好的語言發布了新的版本,一個划時代的大版本: PHP7 。
PHP7 修復了大量 BUG ,新增了功能和語法糖。這些改動涉及到了核心包、 GD庫、 PDO 、 ZIP 、 ZLIB 等熟悉和不熟悉的核心功能與擴展包。
PHP7 移除了已經被廢棄的函數,如 mysql_ 系列函數在 PHP5.5 被廢棄,在 PHP7被刪除。
PHP7 的性能高於 HHVM 。並且是 PHP5.6 的兩倍。
http://php.net/archive/2015.php#id2015-12-03-1
2015 年 12 月 3 號
PHP 開發團隊宣布 PHP 7.0.0 即將上市。本次發布標志着新的重要的 PHP 7 系列的開始。
PHP 7.0.0 附帶了一個新版本的 Zend 引擎中,無數的改進和新功能,如
性能改善: PHP 7 高達兩倍快的 PHP 5.6
顯著減少內存使用
抽象語法樹
一致的 64 位支持
改進的異常層次結構
許多轉化為異常致命錯誤
安全隨機數發生器
刪除舊的和不支持的 SAPIs 和擴展
空合並運算符(?)
返回和標量類型聲明
匿名類
零成本斷言
這是下一個主要版本的 PHP 。它的發布是近兩年的發展征程的結果。這是核心團隊的一個非常特殊的成就。而且,它是許多活躍的社區成員難以置信努力的結果。事實上,這是一個新的 PHP 一代的崛起與巨大潛力。
恭喜大家,這是一個壯觀的 PHP 的世界!
感謝感謝所有的貢獻者和支持者!
根據更新日志,我整理了一下涉及到的類庫: Core 、 CLI_server 、 COM 、 Curl、 Date 、 DBA 、 DOM 、 EXIF 、 Fileinfo 、 Filter 、 FPM 、 FTP 、 GD 、 GMP、 hash 、 IMAP 、 Intl 、 JSON 、 LDAP 、 LiteSpeed 、 libxml 、 Mcrypt 、 Mysqli 、 OCI8 、 ODBC 、 Opcache 、 OpenSSL 、 Pcntl 、 PCRE 、 PDO 、 PDO_DBlib 、 PDO_mysql 、 PDO_OCI 、 PDO_pgsql 、 Phar 、 Phpdbg 、 Reflection 、 Session 、 OAP 、 SPL 、 SQLite3 、 tandard 、 Streams 、 Tokenizer 、 XMLReader 、 XMLRPC 、 XSL 、 Zlib 、 Zip
一、 PHP7 的前世今生
以下摘自並修改與鳥哥微信
PHP7 開始於 2014 年春節,因為基於 PHP-5.5 的 Opcache JIT 因為無法得到期望而擱置了 , 並且讓鳥哥等人認識到 , 基礎部分還不夠好 , 並不能很好的支持 JIT, 所以開始了重構項目 , 希望通過得到 30% 以上的提升。隨后發現性能提升比我們想象的還要大,於是定名為 PHP NG 項目。
經過發起投票 , 絕大部分人都支持了 PHP NG 項目 , 並決定以 PHP NG 為基礎 , 開發新版的 PHP 。社區曾開發過 PHP6 ,后來 PHP6 的特性在 PHP5.5 , 5.6 等版本都逐漸實現,所以 PHP6 被擱置。經過社區投票,新項目命名為 PHP7 。
在這近兩年的時間里,各種新特性的加入 , 性能的持續提升,很多以前不合理的地方改進等等 , 都加入到了 PHP7, 讓 PHP7 越來越豐滿 . 從最底層的 ZVAL 的改變 , 到標量類型提示 , 從最初的 30% 的性能提升 , 到現在超過 100% 的性能飛躍 , 每一處變化都讓人值得期待 . 然后經過幾次不情願的跳票 , 終於 , 到今天 , 這一切都將呈現於你面前。
二、安裝
安裝:我們編譯了核心包以及 PDO , GD , mysqli , Zip 等
>$ ./configure --prefix=/usr/local/php7 --enable-fpm --with-zlib --enable-mbstring --with-openssl --with-mysql --with-mysqli --with-mysql-sock --with-gd --enable-gd-native-ttf --enable-pdo --with-pdo-mysql --with-gettext --with-curl --with-pdo-mysql --enable-sockets --enable-bcmath --enable-xml --with-bz2 --enable-zip
>$ make
>$ sudo make install
三、測試
測試版本:
舊版 PHP 5.5.29 ,新版 PHP 7.0.0
測試機配置如下:
測試性能的方法
class PerformanceTest { private $time; private $memory; public function begin() { $this->time = $this->getTime(); $this->memory = $this->getMemory(); } public function end() { $this->time = $this->getTime() - $this->time; $this->time = round($this->time,7);//在這里才能格式化時間 $this->memory = $this->getMemory() - $this->memory; $this->memory = $this->convert($this->memory); echo "time:{$this->time}秒<br />"; echo "memory:{$this->memory}<br />"; } public function getTime() { list($usec, $sec) = explode(" ", microtime()); return ((float)$usec + (float)$sec); } public function getMemory() { return memory_get_usage(); } public function convert($size) { $unit=array('b','kb','mb','gb','tb','pb'); return @round($size/pow(1024,($i=floor(log($size,1024)))),2).' '.$unit[$i]; } }
1 、測試用例一:
生成五十萬個數組,並查詢五十萬次 key 是否存在
$a = array(); $b = new PerformanceTest(); $b->begin(); for($i=0;$i<500000;$i++){ $a[$i] = $i; } foreach($a as $i) { array_key_exists($i, $a); } $b->end();
/***
php 5.5n
time:0.131秒
memory:40.15 mb
php 7.0n
time:0.0780001秒
memory:14 mb
***/
測試結果如下:
➜ time php test.php
php test.php
0.60s user
0.05s system
98% cpu
0.667 total
➜ time /usr/local/php7/bin/php test.php
/usr/local/php7/bin/php test.php
0.05s user
0.02s system
92% cpu
0.073 total
PHP7 速度是 PHP5.5 的 9 倍
2 、測試用例二:
生成五十萬個數組,並查詢五十萬次 value 是否存在
$a = array(); $b = new PerformanceTest(); $b->begin(); for($i=0;$i<10000;$i++){ $a[$i] = $i; } foreach($a as $i) { in_array($i, $a); } $b->end();
/***
php 5.5n
time:0.6560001秒
memory:845.49 kb
php 7.0n
time:0.095秒
memory:388 kb
***/
➜ time php test.php
php test.php
0.79s user
0.01s system
99% cpu
0.809 total
➜ time /usr/local/php7/bin/php test.php
/usr/local/php7/bin/php test.php
0.08s user
0.01s system
97% cpu
0.091 total
PHP7 速度是 PHP5.5 的 8.7 倍
3 、測試用例三:
示例與結果摘自鳥哥博客。以 Wordpress 為基礎,測試 PHP7 和 HHVM3.2 。用 Apache 的 ab 測試工具。 100 個並發 , 10000 個請求。測試前都會用 100 個請求預熱。
windows下測試命令
php5 和 php7 沒多大差別
ab.exe –n 10000 –c 100 http://localhost/test/test.php
php5.5n
This is ApacheBench, Version 2.3 <$Revision: 1706008 $> Copyright 1996 Adam Twiss, Zeus Technology Ltd, http://www.zeustech.net/ Licensed to The Apache Software Foundation, http://www.apache.org/ Benchmarking localhost (be patient) Server Software: Apache/2.4.18 Server Hostname: localhost Server Port: 80 Document Path: /test/test.php Document Length: 6 bytes Concurrency Level: 100 //100並發 Time taken for tests: 19.447 seconds // 總耗時 Complete requests: 10000 //完成的請求數 Failed requests: 0 Total transferred: 2090000 bytes HTML transferred: 60000 bytes Requests per second: 514.22 [#/sec] (mean) //每秒中完成的請求次數 Time per request: 194.470 [ms] (mean) //每完成100次請求 使用的時間 Time per request: 1.945 [ms] (mean, across all concurrent requests) //每完成一次請求時間 Transfer rate: 104.95 [Kbytes/sec] received Connection Times (ms) min mean[+/-sd] median max Connect: 0 0 0.2 0 1 Processing: 1 186 1351.7 2 14626 Waiting: 0 186 1351.7 1 14625 Total: 1 186 1351.7 2 14626 Percentage of the requests served within a certain time (ms) 50% 2 66% 2 75% 2 80% 2 90% 2 95% 5 98% 2034 99% 12117 100% 14626 (longest request)
PHP7.0
This is ApacheBench, Version 2.3 <$Revision: 1706008 $> Copyright 1996 Adam Twiss, Zeus Technology Ltd, http://www.zeustech.net/ Licensed to The Apache Software Foundation, http://www.apache.org/ Benchmarking localhost (be patient) Server Software: Apache/2.4.18 Server Hostname: localhost Server Port: 80 Document Path: /test/test.php Document Length: 6 bytes Concurrency Level: 100 Time taken for tests: 19.887 seconds Complete requests: 10000 Failed requests: 0 Total transferred: 2080000 bytes HTML transferred: 60000 bytes Requests per second: 502.84 [#/sec] (mean) Time per request: 198.870 [ms] (mean) Time per request: 1.989 [ms] (mean, across all concurrent requests) Transfer rate: 102.14 [Kbytes/sec] received Connection Times (ms) min mean[+/-sd] median max Connect: 0 0 0.2 0 1 Processing: 0 191 1441.4 2 15104 Waiting: 0 191 1441.4 2 15104 Total: 0 191 1441.4 2 15104 Percentage of the requests served within a certain time (ms) 50% 2 66% 3 75% 3 80% 8 90% 9 95% 10 98% 1649 99% 13034 100% 15104 (longest request)
四、新特性
1 、標量類型聲明
有兩種模式 : 強制 ( 默認 ) 和 嚴格模式。 現在可以使用下列類型參數(無論用強制模式還是嚴格模式): 字符串 (string), 整數 (int), 浮點數 (float), 以及布爾值 (bool) 。它們擴充了 PHP5 中引入的其他類型:類名,接口,數組和 回調類型。在舊版中,函數的參數申明只能是 (Array $arr) 、 (CLassName $obj) 等,基本類型比如 Int , String 等是不能夠被申明的
function check(int $bool){ var_dump($bool); } check(2); check(true); check(1.53); //check('rew');//報錯 //check([1,2,3]);//報錯 int(2) int(1) int(1)
若無強制類型轉換,會輸入 int(1)bool(true) 。轉換后會輸出 bool(true) bool(true)
2 、返回值類型聲明
PHP 7 增加了對返回類型聲明的支持。返回類型聲明指明了函數返回值的類型。可用的類型與參數聲明中可用的類型相同。
// ... 3點代表3個參數 而且都是數組 // int 把 bool 轉成 0/1 '123'轉成 123 浮點數 1.23 轉整數 1 數組和對象不可以 function arraysSum(array ...$arrays): array { //array_map(myfunction,array1,array2,array3...) return array_map( //匿名函數 function(array $array): int { return array_sum($array); }, $arrays ); } print_r(arraysSum([1,2,'3','g',[11,22]], [4,5,6], [7,8,9])); /** Array ( [0] => 6 [1] => 15 [2] => 24 ) **/
3 、 null 合並運算符
項目中存在大量同時使用三元表達式和 isset() 的情況,新增了 null 合並運算符 (??)這個語法糖。如果變量存在且值不為 NULL , 它就會返回自身的值,否則返回它的第二個操作數。
$a = ''; $c = isset($a) && !empty($a)? $a : 'err'; $d = $a ?? 'err';//isset($a) && $a != null var_dump($c); var_dump($d); /** string(3) "err" string(0) "" **/
4 、太空船操作符(組合比較符)
太空船操作符用於比較兩個表達式。當 $a 大於、等於或小於 $b 時它分別返回 -1 、0 或 1 。 比較的原則是沿用 PHP 的常規比較規則進行的。
// Integers echo 1 <=> 1; // 0 等於 echo "<br />"; echo 1 <=> 2; // -1 小於 echo "<br />"; echo 2 <=> 1; // 1 大於 echo "<br />"; // Floats echo 1.5 <=> 1.5; // 0 echo "<br />"; echo 1.5 <=> 2.5; // -1 echo "<br />"; echo 2.5 <=> 1.5; // 1 echo "<br />"; // Strings echo "a" <=> "a"; // 0 echo "<br />"; echo "a" <=> "b"; // -1 echo "<br />"; echo "b" <=> "a"; // 1 echo "<br />"; //array echo [1,2,3] <=> [1,2,3]; //0 echo "<br />"; echo [[0,0,0],'a',5,0] <=> [[0,0,0],0.1,1,-1];// -1 首先是數量對比 其次 0位置值對比 如果相等 指針往下 繼續對比 以此類推 字母用到了 accii 但是總比大於0的數值小 小於或等於0的數值大 echo "<br />"; echo [[0,0,0],'a',5,0] <=> [[0,0,0],0.1,1]; //1 echo "<br />";
5 、通過 define() 定義常量數組
define('ANIMALS', ['dog', 'cat', 'bird']); echo ANIMALS[1]; // outputs "cat"
6 、匿名類
現在支持通過 new class 來實例化一個匿名類,這可以用來替代一些“用后即焚”的完整類定義。
interface Logger { public function log(string $msg); } class Application { private $logger; public function getLogger(): Logger { // 返回Logger類 return $this->logger; } public function setLogger(Logger $logger) { $this->logger = $logger; $this->logger->log(12321); } } $app = new Application; $app->setLogger( new class implements Logger { public function log(string $msg) { echo $msg."<br />"; } } ); var_dump($app->getLogger()); /** 12321 object(class@anonymous)#2 (0) { } **/ exit;
7 、 Unicode codepoint 轉譯語法
這接受一個以 16 進制形式的 Unicode codepoint ,並打印出一個雙引號或 heredoc包圍的 UTF-8 編碼格式的字符串。 可以接受任何有效的 codepoint ,並且開頭的 0 是可以省略的。
echo "\u{9876}"; /** 頂 注 舊版直接輸出 **/ exit;
8 、 Closure::call() //調用閉包函數
Closure::call() 現在有着更好的性能,簡短干練的暫時綁定一個方法到對象上閉包並調用它。
//public Closure Closure::bindTo ( object $newthis [, mixed $newscope = 'static' ] ) class Test{ public $name = "lixuan<br/>"; } //PHP7 和 PHP5.6 都可以 $getNameFunc = function(){ return $this->name; }; //echo $getNameFunc();//報錯 $obj1 = new Test; //適用匿名函數 $name = $getNameFunc->bindTo($obj1, 'Test');//把匿名函數綁定(不是添加)到類Test里面 echo $name(); //PHP7 可以 ,PHP5.6 報錯 $getX = function() {return $this->name;}; echo $getX->call(new Test); exit;
9 、為 unserialize() 提供過濾
這個特性旨在提供更安全的方式解包不可靠的數據。它通過白名單的方式來防止潛在的代碼注入。
class A{ public $bb = 'ddd'; } $foo = new A(); var_dump($foo); echo "<br />"; $foo = serialize($foo); var_dump($foo); echo "<br />"; // 將所有對象分為 __PHP_Incomplete_Class 對象 $data = unserialize($foo, ["allowed_classes" => false]); var_dump($data); echo "<br />"; // 將所有對象分為 __PHP_Incomplete_Class 對象 除了 ClassName1 和 ClassName2 $data = unserialize($foo, ["allowed_classes" => ["ClassName1", "ClassName2"]]); var_dump($data); echo "<br />"; // 默認行為,和 unserialize($foo) 相同 $data = unserialize($foo, ["allowed_classes" => true]); var_dump($data); echo "<br />"; /*** object(A)#1 (1) { ["bb"]=> string(3) "ddd" } string(31) "O:1:"A":1:{s:2:"bb";s:3:"ddd";}" object(__PHP_Incomplete_Class)#1 (2) { ["__PHP_Incomplete_Class_Name"]=> string(1) "A" ["bb"]=> string(3) "ddd" } object(__PHP_Incomplete_Class)#2 (2) { ["__PHP_Incomplete_Class_Name"]=> string(1) "A" ["bb"]=> string(3) "ddd" } object(A)#1 (1) { ["bb"]=> string(3) "ddd" } ***/
10 、 IntlChar
新增加的 IntlChar 類旨在暴露出更多的 ICU 功能。這個類自身定義了許多靜態方法用於操作多字符集的 unicode 字符。 Intl 是 Pecl 擴展,使用前需要編譯進 PHP 中,也可 apt-get/yum/port install php5-intl
沒有安裝擴展 沒有進行實際測試
<?php
printf('%x', IntlChar::CODEPOINT_MAX);
echo IntlChar::charName('@');
var_dump(IntlChar::ispunct('!'));
?>
以上例程會輸出:
10ffff
COMMERCIAL AT
bool(true)
11 、預期
預期是向后兼用並增強之前的 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
12 、 Group use declarations
從同一 namespace 導入的類、函數和常量現在可以通過單個 use 語句 一次性導入了。
use 使用到了__autoLoad() 自動加載類(不能自動加載函數 和 常量)
/** spl_autoload_register 自定義自動加載 **/ /* function __autoload($class) { echo $class.":<br />"; $file = $class . '.class.php'; if (is_file($file)) { require_once($file); } } */ function load($class) { //require_once 'lib.php'; echo $class.":<br />"; if(in_array($class,['lib\className','lib\onlyMethod'])) { $file = 'lib\lib.php'; } else { $file = $class . '.class.php'; } echo $file.":<br />"; if (is_file($file)) { require_once($file); } } spl_autoload_register('load');
要使用函數和常量 必須先new 類 自動加載(懶加載)只加載類
require_once('autoLoad.php'); //require_once('lib/lib.php'); //require_once('some\isClass\A.class.php'); use lib\className; use function lib\onlyMethod; /*use some\isClass\A; use some\isClass\B; use some\isClass\C as classC;*/ use some\isClass\{A, B, C as classC}; use some\isFunction\{fn_a, fn_b, fn_c}; /* use function some\isFunction\a; use function some\isFunction\b; use function some\isFunction\c as is_fn_c; */ use function some\isFunction\{a, b, c as is_fn_c}; /* use const some\isFunction\ConstA;//常量命名空間 不適用php define() use const some\isFunction\ConstB; use const some\isFunction\ConstC;*/ use const some\isFunction\{ConstA, ConstB, ConstC}; $a = new A(); echo $a->name."<br />"; $b = new B(); echo $b->name."<br />"; $c = new classC(); echo $c->name."<br />"; new className(); onlyMethod(); //先new類 引進文件 才能使用命名空間函數 new fn_a(); a(); echo "常量A:"; print_r(ConstA); echo "<br />"; new fn_b(); b(); echo "常量B:"; print_r(ConstB); echo "<br />"; new fn_c(); is_fn_c(); echo "常量C:"; print_r(ConstC); echo "<br />";
13 、 intdiv()
接收兩個參數作為被除數和除數,返回他們相除結果的整數部分。
var_dump(intdiv(7, 2)); /** int(3) **/ exit;
14 、 CSPRNG
新增兩個函數 : random_bytes() and random_int(). 可以加密的生產被保護的整數和字符串。我這蹩腳的翻譯,總之隨機數變得安全了。
andom_bytes — 加密生存被保護的偽隨機字符串
random_int — 加密生存被保護的偽隨機整數
$bytes = random_bytes('10'); var_dump(bin2hex($bytes)); //string(20) "3b7257b874bdfd94dd3c" exit;
var_dump(random_int(1, 100)); //int(21) exit; $times = 1000000; $result = [0,0,0,0]; for ($i=0; $i<$times; $i++){ $dieRoll = [1=>0,2=>0,3=>0,4=>0,5=>0,6=>0,]; //initializes just the six counting to zero $dieRoll[roll()] += 1; //first die $dieRoll[roll()] += 1; //second die $dieRoll[roll()] += 1; //third die $result[$dieRoll[6]] += 1; //counts the sixes } function roll(){ return random_int(1,6); } var_dump($result); exit;
15 、 preg_replace_callback_array()
新增了一個函數 preg_replace_callback_array() ,使用該函數可以使得在使用 preg_replace_callback() 函數時代碼變得更加優雅。在 PHP7 之前,回調函數會調用每一個正則表達式,回調函數在部分分支上是被污染了。
$subject = 'AaaCCaaa BbDDb'; preg_replace_callback_array( [ '~[a]+~i' => function ($match) { print_r($match); echo strlen($match[0]), ' matches for "a" found', PHP_EOL; }, '~[b]+~i' => function ($match) { print_r($match); echo strlen($match[0]), ' matches for "b" found', PHP_EOL; } ], $subject ); /*** Array ( [0] => Aaa ) 3 matches for "a" found Array ( [0] => aaa ) 3 matches for "a" found Array ( [0] => Bb ) 2 matches for "b" found Array ( [0] => b ) 1 matches for "b" found ***/ exit;
16 、 Session options
現在, session_start() 函數可以接收一個數組作為參數,可以覆蓋 php.ini 中 session 的配置項。
比如,把 cache_limiter 設置為私有的,同時在閱讀完 session 后立即關閉。
<?php
session_start([
'cache_limiter' => 'private',
'read_and_close' => true,
]);
?>
17 、生成器的返回值
在 PHP5.5 引入生成器的概念。生成器函數每執行一次就得到一個 yield 標識的值。在 PHP7 中,當生成器迭代完成后,可以獲取該生成器函數的返回值。通過 Generator::getReturn() 得到。
function generator() { yield 1; yield 2; yield 3; return "a"; } $generatorClass = ("generator")(); foreach ($generatorClass as $val) { echo $val."<br />"; } echo $generatorClass->getReturn(); /** 1 2 3 a **/ exit;
18 、生成器中引入其他生成器
在生成器中可以引入另一個或幾個生成器,只需要寫 yield from functionName1
function generator1(){ yield 1; yield 2; yield from generator2(); yield from generator3(); } function generator2(){ yield 3; yield 4; } function generator3(){ yield 5; yield 6; } foreach (generator1() as $val){ echo $val, "<br />"; } /*** 1 2 3 4 5 6 ***/ exit;
五、不兼容性(不進行兼容了 就是 "十六進制字符串不再被認為是數字 以前可以兼容")
1 、 foreach 不再改變內部數組指針
在 PHP7 之前,當數組通過 foreach 迭代時,數組指針會移動。現在開始,不再如此,見下面代碼。
$array = [0, 1, 2]; foreach ($array as &$val) { var_dump(current($array)); //next($array); } /** php7 int(0) int(0) int(0) int(0) int(1) int(2)//可以使用next php5 int(1) int(2) bool(false) **/ exit;
2 、 foreach 通過引用遍歷時,有更好的迭代特性
迭代是重復反饋過程的活動,其目的通常是為了逼近所需目標或結果。每一次對過程的重復稱為一次“迭代”,而每一次迭代得到的結果會作為下一次迭代的初始值
當使用引用遍歷數組時,現在 foreach 在迭代中能更好的跟蹤變化。例如,在迭代中添加一個迭代值到數組中,參考下面的代碼:
$array = [0,1,2]; foreach ($array as $key=>&$val) { var_dump($val); $array[$key+1] = 2;//作為下次循環的初始值 if($key >10) { echo "不要死循環<br/>"; break; } } /*** php7 int(0) int(2) int(2) int(2) int(2) int(2) int(2) int(2) int(2) int(2) int(2) int(2) 不要死循環 php5 int(0) int(2) int(2) ***/ exit;
3 、十六進制字符串不再被認為是數字
含十六進制字符串不再被認為是數字
var_dump("0x123" == "291"); var_dump(is_numeric("0x123")); var_dump("0xe" + "0x1"); var_dump(substr("foo", "0x1")); var_dump(0x123 == 291); var_dump(is_numeric(0x123)); var_dump(0xe + 0x1); var_dump(substr("foo", 0x1)); /*** php5 bool(true) bool(true) int(15) string(2) "oo" php7 bool(false) bool(false) int(0) Notice: A non well formed numeric value encountered in E:\WWW\test\test.php on line 9 string(3) "foo" bool(true) bool(true) int(15) string(2) "oo" ***/ exit;
4 、 PHP7 中被移除的函數
被移除的函數列表如下:
call_user_func() 和 call_user_func_array() 從 PHP 4.1.0 開始被廢棄。
已廢棄的 mcrypt_generic_end() 函數已被移除,請使用 mcrypt_generic_deinit() 代替。
/* 打開加密算法和模式 */ $td = mcrypt_module_open('rijndael-256', '', 'ofb', ''); /* 創建初始向量,並且檢測密鑰長度。 * Windows 平台請使用 MCRYPT_RAND。 *MCRYPT_DEV_RANDOM */ $iv = mcrypt_create_iv(mcrypt_enc_get_iv_size($td), MCRYPT_RAND); $ks = mcrypt_enc_get_key_size($td);//32位 echo($iv."<br />"); /* 創建密鑰 */ $key = substr(md5('very secret key'), 0, $ks); echo $key."<br />"; /* 初始化加密 */ mcrypt_generic_init($td, $key, $iv); /* 加密數據 */ $encrypted = mcrypt_generic($td, 'This is very important data'); echo $encrypted."<br />"; /* 結束加密,執行清理工作 */ mcrypt_generic_deinit($td); /* 初始化解密模塊 */ mcrypt_generic_init($td, $key, $iv); /* 解密數據 */ $decrypted = mdecrypt_generic($td, $encrypted); /* 結束解密,執行清理工作,並且關閉模塊 */ mcrypt_generic_deinit($td); mcrypt_module_close($td); /* 顯示文本 */ echo trim($decrypted) . "\n"; /*** ,]P(�Vڈ�{H�LyanF ��Rb5�È a7bc27daf59679de9db7b68b1ef92785 }�^j��"�$�1���IK����}7 This is very important data ***/ exit;
已廢棄的 mcrypt_ecb(), mcrypt_cbc(), mcrypt_cfb() 和 mcrypt_ofb() 函數已被移除。
set_magic_quotes_runtime(), 和它的別名 magic_quotes_runtime() 已被移除 . 它們在 PHP 5.3.0 中已經被廢棄 , 並且 在 in PHP 5.4.0 也由於魔術引號的廢棄而失去功能。
已廢棄的 set_socket_blocking() 函數已被移除,請使用 stream_set_blocking() 代替。
/** 使用方法 bool stream_set_blocking ( resource $stream , bool $mode ) 說明: stream_set_blocking — 為資源流設置阻塞或者阻塞模式 為 stream 設置阻塞或者阻塞模。 此函數適用於支持非阻塞模式的任何資源流(常規文件,套接字資源流等)。 參數說明: stream 資源流。 mode 如果 mode 為0,資源流將會被轉換為非阻塞模式;如果是1,資源流將會被轉換為阻塞模式。 該參數的設置將會影響到像 fgets() 和 fread() 這樣的函數從資源流里讀取數據。 在非阻塞模式下,調用 fgets() 總是會立即返回;而在阻塞模式下,將會一直等到從資源流里面獲取到數據才能返回。 返回值 成功時返回 TRUE, 或者在失敗時返回 FALSE。 這里涉及較多 可以自行查資料 **/
dl() 在 PHP-FPM 不再可用,在 CLI 和 embed SAPIs 中仍可用。
GD 庫中下列函數被移除: imagepsbbox() 、 imagepsencodefont() 、 imagepsextendfont() 、 imagepsfreefont() 、 imagepsloadfont() 、 imagepsslantfont() 、 imagepstext()
在配置文件 php.ini 中, always_populate_raw_post_data 、 asp_tags 、 xsl.security_prefs 被移除了。
5 、 new 操作符創建的對象不能以引用方式賦值給變量
new 操作符創建的對象不能以引用方式賦值給變量
<?php
class C {}
$c =& new C;
?>
PHP5 輸出:
Deprecated: Assigning the return value of new by reference is deprecated in /tmp/test.php on line 3
PHP7 輸出:
Parse error: syntax error, unexpected 'new' (T_NEW) in /tmp/test.php on line 3
6 、移除了 ASP 和 script PHP 標簽
使用類似 ASP 的標簽,以及 script 標簽來區分 PHP 代碼的方式被移除。 受到影響的標簽有: <% %> 、 <%= %> 、 <script language="php"> </script>
7 、從不匹配的上下文發起調用
在不匹配的上下文中以(靜態方式調用非靜態方法), 在 PHP 5.6 中已經廢棄, 但是在 PHP 7.0 中, 會導致被調用方法中未定義 $this 變量,以及此行為已經廢棄的警告。
class A { public function test() { var_dump($this); }//var_dump(123); } // 注意:並沒有從類 A 繼承 class B { public function callNonStaticMethodOfA() { A::test(); } //不是靜態方法 } (new B)->callNonStaticMethodOfA(); /** PHP5 輸出: Deprecated: Non-static method A::test() should not be called statically, assuming $this from incompatible context in /tmp/test.php on line 8 object(B)#1 (0) { } PHP7 輸出: Deprecated: Non-static method A::test() should not be called statically in /tmp/test.php on line 8 Notice: Undefined variable: this in /tmp/test.php on line 3 NULL Deprecated: Non-static method A::test() should not be called statically in E:\WWW\test\test.php on line 15 //反對這么使用 int(123) **/
8 、在數值溢出的時候,內部函數將會失敗
將浮點數轉換為整數的時候,如果浮點數值太大,導致無法以整數表達的情況下,在之前的版本中,內部函數會直接將整數截斷,並不會引發錯誤。 在 PHP 7.0 中,如果發生這種情況,會引發 E_WARNING 錯誤,並且返回 NULL 。
9 、 JSON 擴展已經被 JSOND 取代
JSON 擴展已經被 JSOND 擴展取代。 對於數值的處理,有以下兩點需要注意的:第一,數值不能以點號( . )結束 (例如,數值 34. 必須寫作 34.0 或 34 )。 第二,如果使用科學計數法表示數值, e 前面必須不是點號( . ) (例如, 3.e3 必須寫作3.0e3 或 3e3 )。
10 、 INI 文件中 # 注釋格式被移除
在配置文件 INI 文件中,不再支持以 # 開始的注釋行, 請使用 ; (分號)來表示注釋。 此變更適用於 php.ini 以及用 parse_ini_file() 和 parse_ini_string() 函數來處理的文件。
11 、 $HTTP_RAW_POST_DATA 被移除
不再提供 $HTTP_RAW_POST_DATA 變量。 請使用 php://input 作為替代。
//@file phpinput_server.php $raw_post_data = file_get_contents('php://input', 'r'); echo "-------\$_POST------------------\n"; echo var_dump($_POST) . "\n"; echo "-------php://input-------------\n"; echo $raw_post_data . "\n"; /** -------$_POST------------------ array(2) { ["text1"]=> string(3) "123" ["text2"]=> string(3) "123" } -------php://input------------- text1=123&text2=123 **/ exit;
12 、 yield 變更為右聯接運算符
在使用 yield 關鍵字的時候,不再需要括號, 並且它變更為右聯接操作符,其運算符優先級介於 print 和 => 之間。 這可能導致現有代碼的行為發生改變。可以通過使用括號來消除歧義。
<?php
echo yield -1;
// 在之前版本中會被解釋為:
echo (yield) - 1;
// 現在,它將被解釋為:
echo yield (-1);
yield $foo or die;
// 在之前版本中會被解釋為:
yield ($foo or die);
// 現在,它將被解釋為:
(yield $foo) or die;
?>
function generator() { $foo = 'cc'; yield -1; yield $foo or die; return 'last'; } $generatorClass = ("generator")(); foreach ($generatorClass as $val) { echo $val."<br />"; } echo $generatorClass->getReturn(); /** -1 cc **/ exit;
PHP 官方網站文檔: http://php.net/manual/zh/migration70.php
可以瀏覽 PHP5.6 到 PHP7 時,新特性、新增函數、已經被移除的函數、不兼容性、新增的類和接口等內容。