作者 陈浩 ( 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 目录下 ?>
这样的命名方式被PEAR 、Zend Framework 及各种PHP 项目广泛采用。虽然该方法可以避免不同包或类库中的类名产生冲突,但在书写代码的时候显得较为麻烦和笨拙。
在PHP5.3 中,则只需要指定不同的命名空间即可,命名空间的分隔符为反斜杆/ 。
代码示例:
<?php namespace Zend/Db/Table; class Select {} ?>
这样即使其它命名空间下存在名为Select 的类,程序在调用时也不会产生冲突。代码的可读性也有所增加。
支持延迟静态绑定( Late Static Binding )在PHP5 中,我们可以在类中通过self 关键字或者 __CLASS__ 来判断或调用当前类。但有一个问题,如果我们是在子类中调用,得到的结果将是父类。因为在继承父类的时候,静态成员就已经被绑定了。 例如:
代码示例:
<?php class A { public static function who () { echo __CLASS__ ; } public static function test () { self :: who (); } } class B extends A { public static function who () { echo __CLASS__ ; } } B :: test (); ?>
以上代码输出的结果是: A
这和我们的预期不同,我们原来想得到子类的相应结果。
PHP 5.3.0 中增加了一个static 关键字来引用当前类,即实现了延迟静态绑定:
代码示例:
<?php class A { public static function who () { echo __CLASS__ ; } public static function test () { static:: who (); // 这里实现了延迟的静态绑定 } } class B extends A { 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/intl 、ext/fileinfo 、ext/sqlite3 和ext/enchant 等扩展默认随PHP 绑定发布。其中Phar 可用于打包PHP 程序,类似于Java 中的jar 机制。
6. ereg 正则表达式函数 不再默认可用,请使用速度更快的 PCRE 正则表达式函数
结语:
PHP 5.3 是一个有很大改进的 PHP 版本,但它仍然遵循着 PHP 的设计原则 — 强大、易用。 PHP5.3 一方面在面向对象开发等方面有所加强,使 PHP 更合适于进行企业应用开发,另一方面也增加了不少实用的语法特性和新扩展。我们期待着它能够早日稳定,成为 WEB 开发中的又一个利器。
<END>
发表于 @ 2009年09月25日