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 时,新特性、新增函数、已经被移除的函数、不兼容性、新增的类和接口等内容。