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