PHP5.3新特性

    技术2024-12-27  18

    作者 陈浩 Haohappy

    MSN: haohappy # php.net

    2009-08-13

     

    本文刊于《程序员》杂志 2009.09

    转载请注明作者及出处

     

    毫无疑问,如今 PHP 已经成为 WEB 开方当中最热门的技术之一。根据 nexen.net 的调查,互联网上三分之一的网站选择 PHP 来开发服务器端程序。在欧美和日本等国家, PHP 开发市场呈现出一片欣欣向荣的景象,像 Facebook Yahoo! Flickr Sourceforge 这样的知名 PHP 站点数不胜数。而近年来国内的各大网站也逐渐大量使用 PHP

     

    依靠活跃、组织严密的开发社区, PHP 语言本身一直在稳定地进步 -- 一方面不断改善性能和稳定性,增加各种实用的开发工具;另一方面积极汲取其它编程语言的优点来充实语言特性。如今的 PHP ,即可以支持强大的面向对象开发(如 Java ),又保留了简单易学的语法(如 C ),同时, PHP 还拥有极其多样化的各种实用的函数、扩展和类库,非常方便用于 WEB 开发。另外,随着面向对象开发的逐步普及,各种开源的 PHP 类库和开发框架层出不穷。

     

    6 月底, PHP 官方正式发布了 PHP5.3.0 。这是一个不寻常的 PHP 版本,因为它修复了不少 Bug (超过 140 个),而且带来了很多让 PHP 程序员们期待已久的新特性。其中有些特性原本是计划在 PHP6 中发布的,但因呼声很高,提前在 PHP5.3 中发布了。

     

    让我们来了解一下 PHP5.3 中有哪些好东西吧。

     

    一. PHP 5.3 中的新特性

    支持命名空间 Namespace

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

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

     

     

    代码示例:

    <?php class Zend_Db_Table_Select {} // 表示当前这个类的文件位于Zend/Db/Table/Select 目录下 ?>

    这样的命名方式被PEARZend Framework 及各种PHP 项目广泛采用。虽然该方法可以避免不同包或类库中的类名产生冲突,但在书写代码的时候显得较为麻烦和笨拙。

    PHP5.3 中,则只需要指定不同的命名空间即可,命名空间的分隔符为反斜杆/

     

     

     

    代码示例:

    <?php namespace Zend/Db/Table; class Select {} ?>

    这样即使其它命名空间下存在名为Select 的类,程序在调用时也不会产生冲突。代码的可读性也有所增加。

    支持延迟静态绑定( Late Static Binding

    PHP5 中,我们可以在类中通过self 关键字或者 __CLASS__ 来判断或调用当前类。但有一个问题,如果我们是在子类中调用,得到的结果将是父类。因为在继承父类的时候,静态成员就已经被绑定了。 例如:

     

     

     

    代码示例:

    <?php class  {     public static function  who () {         echo  __CLASS__ ;     }     public static function  test () {          self :: who ();     } } class  extends  {     public static function  who () {          echo  __CLASS__ ;     } } B :: test (); ?>

    以上代码输出的结果是: A

    这和我们的预期不同,我们原来想得到子类的相应结果。

    PHP 5.3.0 中增加了一个static 关键字来引用当前类,即实现了延迟静态绑定:

     

     

     

    代码示例:

    <?php class  {     public static function  who () {         echo  __CLASS__ ;     }     public static function  test () {         static:: who ();  //  这里实现了延迟的静态绑定      } } class  extends  {     public static function  who () {          echo  __CLASS__ ;     } } B :: test (); ?>

    以上代码输出的结果是: B

     

    支持 goto 语句

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

     

     

    代码示例:

    <?php goto  a ; echo  'Foo' ;   a : echo  'Bar' ; for( $i = 0 , $j = 50 $i < 100 $i ++) {   while( $j --) {     if( $j == 17 ) goto  end ;    }   } echo  "i =  $i " ; end : echo  'j hit 17' ; ?>

    支持闭包、 Lambda/Anonymous 函数

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

    PHP 中,我们也可以通过 create_function() 在代码运行时创建函数。但有一个问题:创建的函数仅在运行时才被编译,而不与其它代码同时被编译成执行码,因此我们无法使用类似APC 这样的执行码缓存来提高代码执行效率。

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

     

     

     

    代码示例:

    <?php echo  preg_replace_callback ( '~-([a-z])~' , function ( $match ) {     return  strtoupper ( $match [ 1 ]); },  'hello-world' ); //  输出  helloWorld

     

    $greet  = function( $name ) {      printf ( "Hello %s/r/n" $name ); }; $greet ( 'World' ); $greet ( 'PHP' );

     

    //... 在某个类中

    $callback  =      function ( $quantity $product ) use ( $tax , & $total )         {     $pricePerItem  constant ( __CLASS__  "::PRICE_"  .   strtoupper ( $product ));     $total  += ( $pricePerItem  $quantity ) * ( $tax  1.0 );  }; array_walk ( $products $callback ); ?>

     

     

    新增两个魔术方法 __callStatic() __invoke()

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

     

     

     

    代码示例:

    <?php class  MethodTest  {     public function  __call ( $name $arguments ) {          //  参数  $name  大小写敏感          echo  " 调用对象方法  ' $name ' "               implode ( ' -- ' $arguments ).  "/n" ;     }      /**  PHP 5.3.0 以上版本中本类方法有效   */      public static function  __callStatic ( $name $arguments ) {          //  参数  $name  大小写敏感          echo  " 调用静态方法 ' $name ' "               implode ( ' -- ' $arguments ).  "/n" ;     } } $obj  = new  MethodTest ; $obj -> runTest ( ' 通过对象调用 ' ); MethodTest :: runTest ( ' 静态调用 ' );   // As of PHP 5.3.0 ?>

    以上代码执行后输出如下: 调用对象方法 'runTest' –- 通过对象调用

    调用静态方法 'runTest' –- 静态调用

     

    以函数形式来调用对象时,__invoke() 方法将被自动调用。

     

     

     

    代码示例:

    <?php class  MethodTest  {     public function  __call ( $name $arguments ) {          //  参数  $name  大小写敏感          echo  "Calling object method ' $name ' "               implode ( ', ' $arguments ).  "/n" ;     }      /**  PHP 5.3.0 以上版本中本类方法有效   */      public static function  __callStatic ( $name $arguments ) {          //  参数  $name  大小写敏感          echo  "Calling static method ' $name ' "               implode ( ', ' $arguments ).  "/n" ;     } } $obj  = new  MethodTest ; $obj -> runTest ( 'in object context' ); MethodTest :: runTest ( 'in static context' );   // As of PHP 5.3.0 ?>

    新增Nowdoc 语法,用法和Heredoc 类似,但使用单引号。Heredoc 则需要通过使用双引号来声明。

    Nowdoc 中不会做任何变量解析,非常适合于传递一段PHP 代码。

     

     

     

    代码示例:

    <?php

     

    // Nowdoc 单引号 PHP 5.3 之后支持

    $name  'MyName' ; echo <<<'EOT' My name is "$name". 

    EOT;

     

    // 上面代码输出 My name is "$name". (( 其中变量不被解析 )

    // Heredoc 不加引号

    echo <<<FOOBAR Hello World! FOOBAR;

     

    // 或者 双引号 PHP 5.3 之后支持

    echo <<<"FOOBAR" Hello World! FOOBAR;

     

    ?>

     

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

    代码示例:

    <?php //  静态变量 function  foo () {     static  $bar  = <<<LABEL Nothing in here... LABEL; } //  类成员、常量 class  foo {     const  BAR  = <<<FOOBAR Constant example FOOBAR;     public  $baz  = <<<FOOBAR Property example FOOBAR; } ?>

    在类外也可使用const 来定义常量

    PHP 中定义常量通常是用这种方式:

     

     

    代码示例:

    <?php define ( "CONSTANT" "Hello world." ); ?>

    PHP5.3 新增了一种常量定义方式:

     

     

    代码示例:

    <?php const  CONSTANT  'Hello World' ;

    ?>

    三元运算符增加了一个快捷书写方式: ?:

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

    PHP5.3 新增一种书写方式,可以省略中间部分,书写为 expr1 ?: expr3 如果expr1 结果为True, 则返回expr1 的结果

    HTTP 状态码在200-399 范围内均被认为访问成功 支持动态调用静态方法

    代码示例:

    <?php class  Test {     public static function  testgo ()     {          echo  "gogo!" ;     } } $class  'Test' ; $action  'testgo' ; $class :: $action ();   // 输出  "gogo!" ?>

    支持嵌套处理异常(Exception 新的垃圾收集器(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/intlext/fileinfoext/sqlite3ext/enchant 等扩展默认随PHP 绑定发布。其中Phar 可用于打包PHP 程序,类似于Java 中的jar 机制。

    6.      ereg 正则表达式函数 不再默认可用,请使用速度更快的 PCRE 正则表达式函数

    结语:

    PHP 5.3 是一个有很大改进的 PHP 版本,但它仍然遵循着 PHP 的设计原则 强大、易用。 PHP5.3 一方面在面向对象开发等方面有所加强,使 PHP 更合适于进行企业应用开发,另一方面也增加了不少实用的语法特性和新扩展。我们期待着它能够早日稳定,成为 WEB 开发中的又一个利器。

     

    <END>

    发表于 @ 2009年09月25日

    最新回复(0)