一.
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的结果。
如果expr1结果为True,则返回expr2的结果。
PHP5.3
新增一种书写方式,可以省略中间部分,书写为expr1 ?: expr3
如果expr1结果为True,则返回expr1的结果
如果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开发中的又一个利器。