PHP5.3, PHP5.4, PHP5.5新特性

由于用到PHP新版本,一些新特性必需要了解,且有些能够在开发时就使用,若是不使用,那么何须升级PHP版本呢,显得有些得不偿失了!
因此整理了一下 一些特性,有可能不全,待添加php

PHP 5.3中的新特性html

一.PHP 5.3中的新特性
1. 支持命名空间 (Namespace)
2. 支持延迟静态绑定(Late Static Binding)
3. 支持goto语句
4. 支持闭包、Lambda/Anonymous函数
5. 新增两个魔术方法__callStatic()和__invoke()
6. 新增Nowdoc语法
7. 在类外也可以使用const来定义常量
8. 三元运算符增长了一个快捷书写方式:
9. HTTP状态码在200-399范围内均被认为访问成功
10.支持动态调用静态方法java

1.支持命名空间 (Namespace)
毫无疑问,命名空间是PHP5.3所带来的最重要的新特性。有了命名空间的概念,在开发大型站点时,就比较容易设计出灵活的结构,同时避免不一样包中的类名或变量名产生冲突。mysql

在PHP5.3以前,惯例的划分Package的办法是经过目录名来分隔代码文件,代码中的类名则用下划线_来表示目录。例如web

1 <!--
2 <?php
3 class Zend_Db_Table_Select {}
4 // 表示当前这个类的文件位于Zend/Db/Table/Select目录下
5 ?>
6 -->

这样的命名方式被PEAR、Zend Framework及各类PHP项目普遍采用。虽然该方法能够避免不一样包或类库中的类名产生冲突,但在书写代码的时候显得较为麻烦和笨拙。
在PHP5.3中,则只须要指定不一样的命名空间便可,命名空间的分隔符为反斜杆\。
select.php正则表达式

1 <!--
2 <?php
3 namespace Zend\Db\Table;
4 class Select {}
5 ?>
6 -->

这样即便其它命名空间下存在名为Select的类,程序在调用时也不会产生冲突。代码的可读性也有所增长。
调用方法
call.php算法

1 <!--
2 <?php
3 //namespace Zend\Db;
4 include('select.php');
5 $s new Zend\Db\Table\Select();
6 $s->test();
7 ?>
8 -->

2.支持延迟静态绑定(Late Static Binding)
在PHP5中,咱们能够在类中经过self关键字或者__CLASS__来判断或调用当前类。但有一个问题,若是咱们是在子类中调用,获得的结果将是父类。由于在继承父类的时候,静态成员就已经被绑定了。 例如:sql

01 <!--
02 <?php
03 class A {
04     public static function who() {
05         echo __CLASS__;
06     }
07     public static function test() {
08         self::who();
09     }
10 }
11 class extends A {
12     public static function who() {
13          echo __CLASS__;
14     }
15 }
16 B::test();
17 ?>
18 -->

以上代码输出的结果是:
A
这和咱们的预期不一样,咱们原来想获得子类的相应结果。
PHP 5.3.0中增长了一个static关键字来引用当前类,即实现了延迟静态绑定:编程

01 <!--
02 <?php
03 class A {
04     public static function who() {
05         echo __CLASS__;
06     }
07     public static function test() {
08         static::who(); // 这里实现了延迟的静态绑定
09     }
10 }
11 class extends A {
12     public static function who() {
13          echo __CLASS__;
14     }
15
16  
17 B::test();
18 ?>
19 -->

以上代码输出的结果是:
B
3.支持goto语句
多数计算机程序设计语言中都支持无条件转向语句goto,当程序执行到goto语句时,即转向由goto语句中的标号指出的程序位置继续执行。尽管goto语句有可能会致使程序流程不清晰,可读性减弱,但在某些状况下具备其独特的方便之处,例如中断深度嵌套的循环和 if 语句。json

01 <!--
02 <?php
03 goto a;
04 echo 'Foo';
05 a:
06 echo 'Bar';
07 for($i=0,$j=50; $i<100; $i++) {
08   while($j--) {
09     if($j==17) goto end;
10   }
11 }
12 echo "i = $i";
13 end:
14 echo 'j hit 17';
15 ?>
16 -->

4.支持闭包、Lambda/Anonymous函数
闭包(Closure)函数和Lambda函数的概念来自于函数编程领域。例如JavaScript 是支持闭包和 lambda 函数的最多见语言之一。

在PHP中,咱们也能够经过create_function()在代码运行时建立函数。但有一个问题:建立的函数仅在运行时才被编译,而不与其它代码同时被编译成执行码,所以咱们没法使用相似APC这样的执行码缓存来提升代码执行效率。

在PHP5.3中,咱们可使用Lambda/匿名函数来定义一些临时使用(即用即弃型)的函数,以做为array_map()/array_walk()等函数的回调函数。

01 <!--
02 <?php
03 echo preg_replace_callback('~-([a-z])~'function ($match) {
04     return strtoupper($match[1]);
05 }, 'hello-world');
06 // 输出 helloWorld
07 $greet function($name)
08 {
09     printf("Hello %s\r\n"$name);
10 };
11 $greet('World');
12 $greet('PHP');
13 //...在某个类中
14 $callback =      function ($quantity$productuse ($tax, &$total)         {
15    $pricePerItem = constant(__CLASS__ "::PRICE_" .  strtoupper($product));
16    $total += ($pricePerItem $quantity) * ($tax + 1.0);
17  };
18 array_walk($products$callback);
19 ?>
20 -->

5. 新增两个魔术方法__callStatic()和__invoke()
PHP中本来有一个魔术方法__call(),当代码调用对象的某个不存在的方法时该魔术方法会被自动调用。新增的__callStatic()方法则只用于静态类方法。当尝试调用类中不存在的静态方法时,__callStatic()魔术方法将被自动调用。

01 <!--
02 <?php
03 class MethodTest {
04     public function __call($name$arguments) {
05         // 参数 $name 大小写敏感
06         echo "调用对象方法 '$name' "
07              . implode(' -- '$arguments). "\n";
08     
09  
10     /**  PHP 5.3.0 以上版本中本类方法有效  */
11     public static function __callStatic($name$arguments) {
12         // 参数 $name 大小写敏感
13         echo "调用静态方法 '$name' "
14              . implode(' -- '$arguments). "\n";
15     }
16
17  
18 $obj new MethodTest;
19 $obj->runTest('经过对象调用'); 
20  
21 MethodTest::runTest('静态调用');  // As of PHP 5.3.0
22 ?>
23 -->

以上代码执行后输出以下:
调用对象方法’runTest’ –- 经过对象调用调用静态方法’runTest’ –- 静态调用
以函数形式来调用对象时,__invoke()方法将被自动调用。

01 <!--
02 <?php
03 class MethodTest {
04     public function __call($name$arguments) {
05         // 参数 $name 大小写敏感
06         echo "Calling object method '$name' "
07              . implode(', '$arguments). "\n";
08     
09  
10     /**  PHP 5.3.0 以上版本中本类方法有效  */
11     public static function __callStatic($name$arguments) {
12         // 参数 $name 大小写敏感
13         echo "Calling static method '$name' "
14              . implode(', '$arguments). "\n";
15     }
16
17  
18 $obj new MethodTest;
19 $obj->runTest('in object context'); 
20  
21 MethodTest::runTest('in static context');  // As of PHP 5.3.0
22 ?>
23 -->

6.新增Nowdoc语法
用法和Heredoc相似,但使用单引号。Heredoc则须要经过使用双引号来声明。
Nowdoc中不会作任何变量解析,很是适合于传递一段PHP代码。

01 <!--
02 <?php
03 // Nowdoc 单引号 PHP 5.3以后支持
04 $name 'MyName';
05 echo <<<'EOT'
06 My name is "$name".
07 EOT;
08 //上面代码输出 My name is "$name". ((其中变量不被解析)
09 // Heredoc不加引号
10 echo <<<FOOBAR
11 Hello World!
12 FOOBAR;
13 //或者 双引号 PHP 5.3以后支持
14 echo <<<"FOOBAR"
15 Hello World!
16 FOOBAR;
17 ?>
18 -->

支持经过Heredoc来初始化静态变量、类成员和类常量。

01 <!--
02 <?php
03 // 静态变量
04 function foo()
05 {
06     static $bar = <<<LABEL
07 Nothing in here...
08 LABEL;
09
10  
11 // 类成员、常量
12 class foo
13 {
14     const BAR = <<<FOOBAR
15 Constant example
16 FOOBAR; 
17  
18     public $baz = <<<FOOBAR
19 Property example
20 FOOBAR;
21 }
22 ?>
23 -->

7. 在类外也可以使用const来定义常量
PHP中定义常量一般是用这种方式:

1 <!--
2 <?php
3 define("CONSTANT""Hello world.");
4 ?>
5 -->

而且新增了一种常量定义方式:

1 <!--
2 <?php
3 const CONSTANT = 'Hello World';
4 ?>
5 -->

8. 三元运算符增长了一个快捷书写方式

1 <!--
2 ?:
3 -->

本来格式为是(expr1) ? (expr2) : (expr3)
若是expr1结果为True,则返回expr2的结果。

PHP5.3新增一种书写方式,能够省略中间部分,书写为expr1 ?: expr3
若是expr1结果为True,则返回expr1的结果
9. HTTP状态码在200-399范围内均被认为访问成功
10.支持动态调用静态方法

01 <!--
02 <?php
03 class Test{
04     public static function testgo()
05     {
06          echo "gogo!";
07     }
08 }
09 $class 'Test';
10 $action 'testgo';
11 $class::$action();  //输出 "gogo!"
12 ?>
13 -->

11. 支持嵌套处理异常(Exception)
12. 新的垃圾收集器(GC),并默认启用

二.PHP5.3中其它值得注意的改变
1. 修复了大量bug
2. PHP性能提升
3. php.ini中可以使用变量
4. mysqlnd进入核心扩展 理论上说该扩展访问mysql速度会较以前的MySQL 和 MySQLi 扩展快(参见http://dev.mysql.com/downloads/connector/php-mysqlnd/)
5. ext/phar、ext/intl、ext/fileinfo、ext/sqlite3和ext/enchant等扩展默认随PHP绑定发布。其中Phar可用于打包PHP程序,相似于Java中的jar机制。
6. ereg 正则表达式函数 再也不默承认用,请使用速度更快的PCRE 正则表达式函数


PHP 5.4中的新特性

1. Buid-in web server内置了一个简单的Web服务器
把当前目录做为Root Document只须要这条命令便可:
# php -S localhost:3300
也能够指定其它路径:
# php -S localhost:3300 -t /path/to/root
还能够指定路由:
# php -S localhost:3300 router.php

2.Traits
Traits提供了一种灵活的代码重用机制,即不像interface同样只能定义方法但不能实现,又不能像class同样只能单继承。至于在实践中怎样使用,还须要深刻思考。
魔术常量为__TRAIT__

3. Short array syntax 数组简短语法

1 <!--
2 $arr = [1,'tsing', 'tsingpost.com'];
3 $array = [
4   "foo" => "bar",
5   "bar" => "foo"
6   ];
7 -->

4. Array dereferencing 数组值

01 <!--
02 function myfunc() {
03     return array(1,'tsing', 'tsingpost.com');
04 }
05 我认为比数组简短语法更方便的是dereferencing,之前咱们须要这样:
06 $arr = myfunc();
07 echo $arr[1];
08 在PHP5.4中这样就好了:
09 echo myfunc()[1];
10 其余:
11 $name = explode(",", "tsings,male")[0];
12 explode(",", "tsings,male")[3] = "phper";
13 -->

5. Upload progress

Session提供了上传进度支持,经过$_SESSION["upload_progress_name"]就能够得到当前文件上传的进度信息,结合Ajax就能很容易实现上传进度条了。

6. JsonSerializable Interface
实现了JsonSerializable接口的类的实例在json_encode序列化的以前会调用jsonSerialize方法,而不是直接序列化对象的属性。

7. Use mysqlnd by default

1 <!--
2 如今mysql, mysqli, pdo_mysql默认使用mysqlnd本地库,在PHP5.4之前须要:
3 $./configure --with-mysqli=mysqlnd
4 如今:
5 $./configure --with-mysqli
6 -->

8.实例化类

1 <!--
2 class test{
3     function show(){
4     return 'test';
5     }
6 }
7 echo (new test())->show();
8 -->

9.支持 Class::{expr}() 语法

1 <!--
2 foreach ([new Human("Gonzalo"), new Human("Peter")] as $human) {
3     echo $human->{'hello'}();
4 }
5 -->

10.Callable typehint

01 <!--
02 function foo(callable $callback) {
03 }
04   则:
05   foo("false"); //错误,由于false不是callable类型
06   foo("printf"); //正确
07   foo(function(){}); //正确
08 class A {
09   static function show() {
10     }
11 }
12   foo(array("A", "show")); //正确
13 -->

11.函数类型提示的加强

01 <!--
02 因为php是弱类型的语言,所以在php 5.0后,引入了函数类型提示的功能,其含义为对于传入函数中的参数都进行类型检查,举个例子,有以下的类:
03 class bar {
04 function foo(bar $foo) {
05 }
06 //其中函数foo中的参数规定了传入的参数必须为bar类的实例,不然系统会判断出错。一样对于数组来讲,也能够进行判断,好比:
07 function foo(array $foo) {
08 }
09 }
10   foo(array(1, 2, 3)); // 正确,由于传入的是数组
11   foo(123); // 不正确,传入的不是数组
12 -->

12.新增长了$_SERVER["REQUEST_TIME_FLOAT"],这个是用来统计服务请求时间的,并用ms来表示

1 <!--
2 echo "脚本执行时间 ", round(microtime(true) - $_SERVER["REQUEST_TIME_FLOAT"], 2), "s";
3  -->

13. 让Json更懂中文(JSON_UNESCAPED_UNICODE)

1 <!--
2 echo json_encode("中文", JSON_UNESCAPED_UNICODE);
3 //"中文"
4 -->

14. 二进制直接量(binary number format)

1 <!--
2 $bin  = 0b1101;
3 echo $bin;
4 //13
5 -->

PHP 5.4.0 性能大幅提高, 修复超过100个bug.
废除了register_globals, magic_quotes以及安全模式。
另外值得一提的是多字节支持已经默认启用了,
default_charset从ISO-8859-1已经变为UTF-8.
默认发送“Content-Type: text/html; charset=utf-8”,
你不再须要在HTML里写meta tag,也无需为UTF-8兼容而传送额外的header了。

删除的特性
最后,咱们集中整理了几年来标记为已弃用的多个特性。这些特性包括 allow_call_time_pass_reference、define_syslog_variables、highlight.bg、register_globals、register_long_arrays、magic_quotes、safe_mode、zend.ze1_compatibility_mode、session.bug_compat4二、session.bug_compat_warn 以及 y2k_compliance。
除了这些特性以外,magic_quotes 多是最大的危险。在早期版本中,未考虑因 magic_quotes 出错致使的后果,简单编写且未采起任何举措使自身免受 SQL 注入攻击的应用程序都经过 magic_quotes 来保护。若是在升级到 PHP 5.4 时未验证已采起正确的 SQLi 保护措施,则可能致使安全漏洞。

其余改动和特性
有一种新的“可调用的”类型提示,用于某方法采用回调做为参数的状况。
htmlspecialchars() 和 htmlentities() 如今可更好地支持亚洲字符,若是未在 php.ini 文件中显式设置 PHP default_charset,这两个函数默认使用 UTF-8 而不是 ISO-8859-1。
会话 ID 如今默认经过 /dev/urandom(或等效文件)中的熵生成,而不是与早期版本同样成为必须显式启用的一个选项。
mysqlnd 这一捆绑的 MySQL 原生驱动程序库如今默认用于与 MySQL 通讯的各类扩展,除非在编译时经过 ./configure 被显式覆盖。
可能还有 100 个小的改动和特性。从 PHP 5.3 升级到 5.4 应该极为顺畅,但请阅读迁移指南加以确保。若是您从早期版本升级,执行的操做可能稍多一些。请查看之前的迁移指南再开始升级。


PHP 5.5中的新特性

新特性及提案列表 都至关大,并且不是按重要性排序。因此,若是你不想通读一遍的话,这里有四个特色我我的是最兴奋的:
:一个简单的密码散列API
:标量类型提示
:Getter和Setter   
:生成器
如今,让咱们来看看PHP5.5 可能会新增的功能:

一、放弃对Windows XP和2003 的支持

2.弃用e修饰符
e修饰符是指示preg_replace函数用来评估替换字符串做为PHP代码,而不仅是仅仅作一个简单的字符串替换。不出所料,这种行为会源源不断的出现安全问题。这就是为何在PHP5.5 中使用这个修饰符将抛出一个弃用警告。做为替代,你应该使用preg_replace_callback函数。你能够从RFC找到更多关于这个变化相应的信息。

3.新增函数和类

01 <!--
02 boolval()
03 PHP已经实现了strval、intval和floatval的函数。为了达到一致性将添加boolval函数。它彻底能够做为一个布尔值计算,也能够做为一个回调函数。  
04  
05 hash_pbkdf2()
06 PBKDF2全称“Password-Based Key Derivation Function 2”,正如它的名字同样,是一种从密码派生出加密密钥的算法。这就须要加密算法,也能够用于对密码哈希。更普遍的说明和用法示例 
07  
08 array_column()
09 $userNames = array_column($users, 'name');
10 // is the same as
11 $userNames = [];
12 foreach ($users as $user) {
13     $userNames[] = $user['name'];
14
15  
16 intl 扩展
17 将有许多改进 intl的扩展。例如,将会有新的IntlCalendar,IntlGregorianCalendar,IntlTimeZone,IntlBreakIterator,IntlRuleBasedBreakIterator,IntlCodePointBreakIterator类。以前,我居然不知道有这么多关于intl扩展,若是你想知道更多,我建议你去最新公告里找 Calendar和 BreakIterator。
18 -->

4.一个简单的密码散列API

01 <!--
02 $password = "foo";
03 // creating the hash
04 $hash = password_hash($password, PASSWORD_BCRYPT);
05 // verifying a password
06 if (password_verify($password, $hash)) {
07     // password correct!
08 } else {
09     // password wrong!
10 }
11 -->

5.新的语言特性和加强功能。

01 <!--
02 常量引用
03 “常量引用”意味着数组能够直接操做字符串和数组字面值。举两个例子:
04 function randomHexString($length) {
05     $str = '';
06     for ($i = 0; $i < $length; ++$i) {
07         $str .= "0123456789abcdef"[mt_rand(0, 15)]; // direct dereference of string
08     }
09 }
10 function randomBool() {
11     return [false, true][mt_rand(0, 1)]; // direct dereference of array
12 }
13 我不认为在实践中会使用此功能,但它使语言更加一致。请参阅 RFC。
14 -->

6.调用empty()函数(和其余表达式)一块儿工做
目前,empty()语言构造只能用在变量,而不能在其余表达式。
在特定的代码像empty($this->getFriends())将会抛出一个错误。做为PHP5.5 这将成为有效的代码

7.获取完整类别名称

1 <!--
2 PHP5.3 中引入命名空间的别名类和命名空间短版本的功能。虽然这并不适用于字符串类名称
3 use Some\Deeply\Nested\Namespace\FooBar;
4 // does not work, because this will try to use the global FooBar class
5 $reflection = new ReflectionClass('FooBar');
6 echo FooBar::class;
7 为了解决这个问题采用新的FooBar::class语法,它返回类的完整类别名称
8  -->

8.参数跳跃

1 <!--
2 若是你有一个函数接受多个可选的参数,目前没有办法只改变最后一个参数,而让其余全部参数为默认值。
3 RFC上的例子,若是你有一个函数以下:
4 function create_query($where, $order_by, $join_type='', $execute = false, $report_errors = true) { ... }
5 那么有没有办法设置$report_errors=false,而其余两个为默认值。为了解决这个跳跃参数的问题而提出:
6 create_query("deleted=0", "name", default, default, false);
7 我我的不是特别喜欢这个提议。在个人眼睛里,代码须要这个功能,只是设计不当。函数不该该有12个可选参数。
8  -->

9.标量类型提示

01 <!--
02 标量类型提示本来计划进入5.4,但因为缺少共识而没有作。获取更多关于为何标量类型提示没有作进PHP的信息,请参阅: 标量类型提示比你认为的更难。
03 对于PHP5.5 而言,针对标量类型提示讨论又一次出现,我认为这是一个至关不错的 提议。
04 它须要经过输入值来指定类型。例如:123,123.0,“123”都是一个有效的int参数输入,但“hello world”就不是。这与内部函数的行为一致。
05 function foo(int $i) { ... }
06 foo(1);      // $i = 1
07 foo(1.0);    // $i = 1
08 foo("1");    // $i = 1
09 foo("1abc"); // not yet clear, maybe $i = 1 with notice
10 foo(1.5);    // not yet clear, maybe $i = 1 with notice
11 foo([]);     // error
12 foo("abc");  // error
13 -->


11.生成器

01 <!--
02 目前,自定义迭代器不多使用,由于它们的实现,须要大量的样板代码。生成器解决这个问题,并提供了一种简单的样板代码来建立迭代器。
03 例如,你能够定义一个范围函数做为迭代器:
04 <?php
05 function *xrange($start, $end, $step = 1) {
06     for ($i = $start; $i < $end; $i += $step) {
07         yield $i;
08     }
09 }
10 foreach (xrange(10, 20) as $i) {
11     // ...
12 }
13 上述xrange函数具备与内建函数相同的行为,但有一点区别:不是返回一个数组的全部值,而是返回一个迭代器动态生成的值。
14  -->

12.列表解析和生成器表达式

01 <!--
02 列表解析提供一个简单的方法对数组进行小规模操做:
03 $firstNames = [foreach ($users as $user) yield $user->firstName];
04 上述列表解析相等于下面的代码:
05 $firstNames = [];
06 foreach ($users as $user) {
07     $firstNames[] = $user->firstName;
08 }
09 也能够这样过滤数组:
10 $underageUsers = [foreach ($users as $user) if ($user->age < 18) yield $user];
11 生成器表达式也很相似,可是返回一个迭代器(用于动态生成值)而不是一个数组。
12 -->

13.finally关键字

1 <!--
2 这个和java中的finally同样,经典的try ... catch ... finally 三段式异常处理。
3  -->

14.foreach 支持list()

01 <!--
02 对于“数组的数组”进行迭代,以前须要使用两个foreach,如今只须要使用foreach + list了,可是这个数组的数组中的每一个数组的个数须要同样。看文档的例子一看就明白了。
03 $array = [
04     [1, 2],
05     [3, 4],
06 ];
07 foreach ($array as list($a, $b)) {
08     echo "A: $a; B: $b\n";
09 }
10  -->

15.增长了opcache扩展
使用opcache会提升php的性能,你能够和其余扩展同样静态编译(–enable-opcache)或者动态扩展(zend_extension)加入这个优化项。

16.非变量array和string也能支持下标获取了

1 <!--
2 echo array(1, 2, 3)[0];
3 echo [1, 2, 3][0];
4 echo "foobar"[2];
5 -->
相关文章
相关标签/搜索