PHP7新特性一覽及測試實例


1
0

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 –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 時,新特性、新增函數、已經被移除的函數、不兼容性、新增的類和接口等內容。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM