1 PHP 5.3中的新 特性 1.1 支持命名空间 (Namespace) 毫无疑问,命名空间是PHP5.3所带来的最重要的新 特性 。 在PHP5.3中,则只需要指定不同的命名空间即可,命名空间的分隔符为反斜杆\ 。 //select.php ? php namespace Zend\Db\Table; class Select { }
毫无疑问,命名空间是PHP5.3所带来的最重要的新特性。
在PHP5.3中,则只需要指定不同的命名空间即可,命名空间的分隔符为反斜杆\。
//select.php<br><span>php namespace Zend\Db\Table; </span><span>class</span><span> Select { } <br>?> </span>
这样即使其它命名空间下存在名为Select的类,程序在调用时也不会产生冲突。代码的可读性也有所增加。
调用方法:
//call.<span>php </span><span>php </span><span>//</span><span>namespace Zend\Db; </span> <span>include</span>('select.php'<span>); </span><span>$s</span> = <span>new</span><span> Zend\Db\Table\Select(); </span><span>$s</span>->test(); <br>?>
在PHP5中,我们可以在类中通过self关键字或者__CLASS__来判断或调用当前类。但有一个问题,如果我们是在子类中调用,得到的结果将是父类。因为在继承父类的时候,静态成员就已经被绑定了。 例如:
<span>php </span><span>class</span><span> A { </span><span>public</span> <span>static</span> <span>function</span><span> who() { </span><span>echo</span> <span>__CLASS__</span><span>; } </span><span>public</span> <span>static</span> <span>function</span><span> test() { self</span>::<span>who(); } } </span><span>class</span> B <span>extends</span><span> A { </span><span>public</span> <span>static</span> <span>function</span><span> who() { </span><span>echo</span> <span>__CLASS__</span><span>; } } B</span>::test(); <br><br>//以上代码输出的结果是: A ; 这和我们的预期不同,我们原来想得到子类的相应结果。
PHP 5.3.0中增加了一个static关键字来引用当前类,即实现了延迟静态绑定:
<span>php </span><span>class</span><span> A { </span><span>public</span> <span>static</span> <span>function</span><span> who() { </span><span>echo</span> <span>__CLASS__</span><span>; } </span><span>public</span> <span>static</span> <span>function</span><span> test() { </span><span>static</span>::who(); <span>//</span><span> 这里实现了延迟的静态绑定 </span> <span> } } </span><span>class</span> B <span>extends</span><span> A { </span><span>public</span> <span>static</span> <span>function</span><span> who() { </span><span>echo</span> <span>__CLASS__</span><span>; } } B</span>::test(); <br><br>//以上代码输出的结果是: B
多数计算机程序设计语言中都支持无条件转向语句goto,当程序执行到goto语句时,即转向由goto语句中的标号指出的程序位置继续执行。尽管 goto语句有可能会导致程序流程不清晰,可读性减弱,但在某些情况下具有其独特的方便之处,例如中断深度嵌套的循环和 if 语句。
<span>php <br> goto a; </span><span>echo</span> 'Foo'<span>; a</span>: <span>echo</span> 'Bar'<span>; </span><span>for</span>(<span>$i</span>=0,<span>$j</span>=50; <span>$i</span>$i++<span>) { </span><span>while</span>(<span>$j</span>--<span>) { </span><span>if</span>(<span>$j</span>==17) goto <span>end</span><span>; } } </span><span>echo</span> "i = <span>$i</span>"<span>; </span><span>end</span>: <span>echo</span> 'j hit 17';
闭包(Closure)函数和Lambda函数的概念来自于函数编程领域。例如JavaScript 是支持闭包和 lambda 函数的最常见语言之一。 在PHP中,我们也可以通过create_function()在代码运行时创建函数。但有一个问题:创建的函数仅在运行时才被编译,而不与其它代码同时被编译成执行码,因此我们无法使用类似APC这样的执行码缓存来提高代码执行效率。在PHP5.3中,我们可以使用Lambda/匿名函数来定义一些临时使用(即用即弃型)的函数,以作为array_map()/array_walk()等函数的回调函数。
<span>php </span><span>echo</span> <span>preg_replace_callback</span>('~-([a-z])~', <span>function</span> (<span>$match</span><span>) { </span><span>return</span> <span>strtoupper</span>(<span>$match</span>[1<span>]); }</span>, 'hello-world'<span>); </span><span>//</span><span> 输出 helloWorld <br> </span> <span>$greet</span> = <span>function</span>(<span>$name</span><span>) { </span><span>printf</span>("Hello %s\r\n", <span>$name</span><span>); }; </span><span>$greet</span>('World'<span>); </span><span>$greet</span>('PHP'<span>); <br> </span><span>//</span><span>...在某个类中 </span> <span>$callback</span> = <span>function</span> (<span>$quantity</span>, <span>$product</span>) <span>use</span> (<span>$tax</span>, &<span>$total</span><span>) { </span><span>$pricePerItem</span> = <span>constant</span>(<span>__CLASS__</span> . "::PRICE_" . <span>strtoupper</span>(<span>$product</span><span>)); </span><span>$total</span> += (<span>$pricePerItem</span> * <span>$quantity</span>) * (<span>$tax</span> + 1.0<span>); }; </span><span>array_walk</span>(<span>$products</span>, <span>$callback</span>);
PHP中原本有一个魔术方法__call(),当代码调用对象的某个不存在的方法时该魔术方法会被自动调用。新增的__callStatic()方法则只用于静态类方法。当尝试调用类中不存在的静态方法时,__callStatic()魔术方法将被自动调用。
<span>php </span><span>class</span><span> MethodTest { </span><span>public</span> <span>function</span> __call(<span>$name</span>, <span>$arguments</span><span>) { </span><span>//</span><span> 参数 $name 大小写敏感 </span> <span>echo</span> "调用对象方法 '<span>$name</span>' " . <span>implode</span>(' -- ', <span>$arguments</span>). "\n"<span>; } </span><span>/*</span><span>* PHP 5.3.0 以上版本中本类方法有效 </span><span>*/</span> <span>public</span> <span>static</span> <span>function</span> __callStatic(<span>$name</span>, <span>$arguments</span><span>) { </span><span>//</span><span> 参数 $name 大小写敏感 </span> <span>echo</span> "调用静态方法 '<span>$name</span>' " . <span>implode</span>(' -- ', <span>$arguments</span>). "\n"<span>; } } </span><span>$obj</span> = <span>new</span><span> MethodTest; </span><span>$obj</span>->runTest('通过对象调用'<span>); MethodTest</span>::runTest('静态调用'); <span>//</span><span> As of PHP 5.3.0 <br><br>//以上代码执行后输出如下: <br>//调用对象方法'runTest' –- 通过对象调用<br>//调用静态方法'runTest' –- 静态调用<br></span>
以函数形式来调用对象时,__invoke()方法将被自动调用。
<span>php </span><span>class</span><span> MethodTest { </span><span>public</span> <span>function</span> __call(<span>$name</span>, <span>$arguments</span><span>) { </span><span>//</span><span> 参数 $name 大小写敏感 </span> <span>echo</span> "Calling object method '<span>$name</span>' " . <span>implode</span>(', ', <span>$arguments</span>). "\n"<span>; } </span><span>/*</span><span>* PHP 5.3.0 以上版本中本类方法有效 </span><span>*/</span> <span>public</span> <span>static</span> <span>function</span> __callStatic(<span>$name</span>, <span>$arguments</span><span>) { </span><span>//</span><span> 参数 $name 大小写敏感 </span> <span>echo</span> "Calling static method '<span>$name</span>' " . <span>implode</span>(', ', <span>$arguments</span>). "\n"<span>; } } </span><span>$obj</span> = <span>new</span><span> MethodTest; </span><span>$obj</span>->runTest('in object context'<span>); MethodTest</span>::runTest('in static context'); <span>//</span><span> As of PHP 5.3.0 </span>
用法和Heredoc类似,但使用单引号。Heredoc则需要通过使用双引号来声明。 Nowdoc中不会做任何变量解析,非常适合于传递一段PHP代码。
<span>php <span>//<span> Nowdoc 单引号 PHP 5.3之后支持 <span>$name = 'MyName'<span>; <span>echo My name is "<span>$name".<span> EOT; <span>//<span>上面代码输出 My name is "$name". ((其中变量不被解析) // Heredoc不加引号 <span>echo FOOBAR Hello World!<span> FOOBAR; <span>//<span>或者 双引号 PHP 5.3之后支持 <span>echo Hello World!<span> FOOBAR;</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span>
支持通过Heredoc来初始化静态变量、类成员和类常量。
<span>php </span><span>//</span><span> 静态变量 </span> <span>function</span><span> foo() { </span><span>static</span> <span>$bar</span> = LABEL Nothing in here...<span> LABEL; } </span><span>//</span><span> 类成员、常量 </span> <span>class</span><span> foo { </span><span>const</span> BAR = FOOBAR <span>Constant</span><span> example FOOBAR; </span><span>public</span> <span>$baz</span> = FOOBAR Property example FOOBAR; }
<span>//</span><span>PHP中定义常量通常是用这种方式 </span> <span>define</span>("CONSTANT", "Hello world."<span>); </span><span>//</span><span>并且新增了一种常量定义方式 </span> <span>const</span> <span>CONSTANT</span> = 'Hello World';
原本格式为是(expr1) ? (expr2) : (expr3) ; 如果expr1结果为True,则返回expr2的结果。
PHP5.3新增一种书写方式,可以省略中间部分,书写为expr1 ?: expr3 ; 如果expr1结果为True,则返回expr1的结果
<span>//</span><span>原格式 </span> <span>$expr</span>=<span>$expr1</span>?<span>$expr1</span>:<span>$expr2</span> <span>//</span><span>新格式 </span> <span>$expr</span>=<span>$expr1</span>?:<span>$expr2</span>
<span>class</span><span> Test{ </span><span>public</span> <span>static</span> <span>function</span><span> testgo() { </span><span>echo</span> "gogo!"<span>; } } </span><span>$class</span> = 'Test'<span>; </span><span>$action</span> = 'testgo'<span>; </span><span>$class</span>::<span>$action</span>(); <span>//</span><span>输出 "gogo!" </span>
1.1 修复了大量bug
1.2 PHP性能提高
1.3 php.ini中可使用变量
1.4 mysqlnd进入核心扩展 理论上说该扩展访问mysql速度会较之前的MySQL 和 MySQLi 扩展快(参见http://dev.mysql.com/downloads/connector/php-mysqlnd/)
1.5 ext/phar、ext/intl、ext/fileinfo、ext/sqlite3和ext/enchant等扩展默认随PHP绑定发布。其中Phar可用于打包PHP程序,类似于Java中的jar机制。
1.6 ereg 正则表达式函数 不再默认可用,请使用速度更快的PCRE 正则表达式函数
PHP 5.3.0 新增了两个错误等级: E_DEPRECATED 和 E_USER_DEPRECATED. 错误等级
E_DEPRECATED 被用来说明一个函数或者功能已经被弃用. E_USER_DEPRECATED 等级目的在于表明用户代码中的弃用功能,
类似于 E_USER_ERROR 和 E_USER_WARNING 等级.
下面是被弃用的 INI 指令列表. 使用下面任何指令都将导致 E_DEPRECATED 错误.
define_syslog_variables
register_globals
register_long_arrays
safe_mode
magic_quotes_gpc
magic_quotes_runtime
magic_quotes_sybase
弃用 INI 文件中以 '#' 开头的注释.
弃用函数:
call_user_method() (使用 call_user_func() 替代)
call_user_method_array() (使用 call_user_func_array() 替代)
define_syslog_variables()
dl()
ereg() (使用 preg_match() 替代)
ereg_replace() (使用 preg_replace() 替代)
eregi() (使用 preg_match() 配合 'i' 修正符替代)
eregi_replace() (使用 preg_replace() 配合 'i' 修正符替代)
set_magic_quotes_runtime() 以及它的别名函数 magic_quotes_runtime()
session_register() (使用 $_SESSION 超全部变量替代)
session_unregister() (使用 $_SESSION 超全部变量替代)
session_is_registered() (使用 $_SESSION 超全部变量替代)
set_socket_blocking() (使用 stream_set_blocking() 替代)
split() (使用 preg_split() 替代)
spliti() (使用 preg_split() 配合 'i' 修正符替代)
sql_regcase()
mysql_db_query() (使用 mysql_select_db() 和 mysql_query() 替代)
mysql_escape_string() (使用 mysql_real_escape_string() 替代)
废弃以字符串传递区域设置名称. 使用 LC_* 系列常量替代.
mktime() 的 is_dst 参数. 使用新的时区处理函数替代.
弃用的功能:
弃用通过引用分配 new 的返回值.
调用时传递引用被弃用.