毫无疑问,命名空间是PHP5.3所带来的最重要的新特性。
在PHP5.3中,可以用命名空间防止代码的冲突,命名空间的分隔符为 \ 反斜线。
在PHP中,我们可以在类中通过self关键字或者CLASS来判断或调用当前类。但有一个问题,如果我们是在子类中调用,得到的结果将是父类。因为在继承父类的时候,静态成员就已经被绑定了。例如:
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。这和我们的预期结果不同。
PHP5.3中增加了一个static关键字来引用当前类,即实现了延迟静态绑定:
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语句有可能会导致程序流程不清晰,可读性减弱,但在某些情况下具有其独特的方便之处,例如中断深度嵌套的循环和 if 语句。
goto test;
echo '1';
test:
echo '2';
?>
// 运行时会输出 2
闭包(Closure)函数和Lambda函数的概念来自于函数编程领域。例如JavaScript 是支持闭包和 lambda 函数的最常见语言之一。
在PHP5.3中,可以使用匿名函数或Lambda函数来定义一些临时使用(即用即弃型)的函数,以作为array_map()或array_walk()等函数的回调函数。
PHP中原本有一个魔术方法__call(),当调用类中某个不存在的方法时该魔术方法会被自动调用。新增的__callStatic()方法则只用于类的静态方法。当尝试调用类中不存在的静态方法时,__callStatic()魔术方法将被自动调用。
此外,PHP5.3中还新增了__invoke()魔术方法。当尝试以调用函数的方式调用一个对象时,__invoke() 方法会被自动调用。 如:
class CallableClass
{
function __invoke($x) {
var_dump($x);
}
}
$obj = new CallableClass;
$obj(5);
var_dump(is_callable($obj));
?>
/* 输出结果:
int(5)
bool(true)
*/
就象 Heredoc 结构类似于双引号字符串,Nowdoc 结构是类似于单引号字符串的。Nowdoc 结构很象 heredoc 结构,但是 nowdoc 中不进行解析操作。故非常适合于传递一段PHP代码。
$str = <<<'EOD'
Example of string
spanning multiple lines
using nowdoc syntax.
EOD;
echo $str;
//PHP中定义常量通常是用这种方式
define("CONSTANT", "Hello world.");
//PHP5.3中的const关键字也支持在类外定义常量
const CONSTANT = 'Hello World';
原本格式为 expr1 ? expr2 : expr3 。
如果expr1结果为True,则返回expr2的结果,否则返回expr3。
PHP5.3新增了一种书写方式,可以省略中间部分,简写为 expr1 ?: expr3 ,当 expr1 为 TRUE 时返回 expr1,否则返回 expr3。
class Test{
public static function testgo()
{
echo "gogo!";
}
}
$class = 'Test';
$action = 'testgo';
$class::$action(); //输出 "gogo!"
class MyException extends Exception { }
class Test {
public function testing() {
try {
try {
throw new MyException('foo!');
} catch (MyException $e) {
/* rethrow it */
throw $e;
}
} catch (Exception $e) {
return $e->getMessage();
}
}
}
$foo = new Test;
echo $foo->testing();
// 输出:foo!
?>
此外,PHP5.3还弃用了一些老版本的功能和函数,并用新的函数进行替代,这里不做赘述。
Traits提供了一种灵活的代码重用机制,既不像interface一样只能定义方法但不能实现,又不像class一样只能单继承。
// Traits不能被单独实例化,只能被类所包含
trait SayWorld
{ public function sayHello()
{
echo 'World!';
}
}
class MyHelloWorld
{
// 将SayWorld中的成员包含进来
use SayWorld;
}
$xxoo = new MyHelloWorld();
// sayHello() 函数是来自 SayWorld 构件的
$xxoo->sayHello();
// 原来的数组写法
$arr = array("key" => "value", "key2" => "value2");
$arr = array(1,2,3,4);
// 简写形式
$arr = ["key" => "value", "key2" => "value2"];
$arr = [1,2,3,4];
function myfunc() {
return array('jack','rose');
}
// 以前的写法
$arr = myfunc();
echo $arr[0];
// PHP5.4中,支持这样书写
echo myfunc()[0];
//启动Web服务器
php -S localhost:8000
//启动时指定根目录
php -S localhost:8000 -t /home/me/public_html/foo
//使用路由(Router)脚本
php -S localhost:8000 index.php //所有的请求都会由index.php来处理。
(new Foo)->bar();
通过 $_SESSION[“upload_progress_name”] 就可以获得当前文件上传的进度信息,结合 Ajax 就能很容易的实现上传进度条。
PHP5.4中的json_encode()函数的第二个参数,新增了一个常量JSON_UNESCAPED_UNICODE,可有效防止中文字符被转成Unicode编码格式。
echo json_encode(array("中文"));
// 输出:["\u4e2d\u6587"]
echo json_encode(array("中文"), JSON_UNESCAPED_UNICODE);
// 输出:["中文"]
// 如果第二个参数指定为JSON_UNESCAPED_UNICODE,就不会将中文转换为Unicode编码格式
默认发送“Content-Type: text/html; charset=utf-8”,你再也不需要在HTML里写meta 标签了,也无需为UTF-8兼容而传送额外的header了。
$bin = bindec('1101'); //之前需要这样写
$bin = 0b1101;
echo $bin; // 输出13
function foo(array $arr) { // 声明foo函数的参数类型必须是array
echo json_encode($arr);
}
// foo(1); // 这里会报错,foo的参数类型必须是array
foo(array(1,2,3)); // 正确
此外,PHP5.4 废除了register_globals、 magic_quotes、 allow_call_time_pass_reference以及安全模式等等。
在早期版本中,你可以在函数调用时,在参数前添加&修饰符来指明参数变量按引用传递。但在 PHP 5.4 中,该用法已被移除,你只需要在函数声明时指定按引用传递即可。
// 早期版本中,要想给一个函数传递一个引用,是这样写的
/* function foo($a) {
$a = 100;
}
$a = 3;
foo(&$a);
echo $a; */
// 自PHP5.4以后,不能再像上面那样用,如果还想给一个函数传递一个引用,可以在声明时就指明
function goo(&$a) {
$a = 100;
}
$a = 3;
goo($a);
echo $a; // 输出:100
yield关键字用于当函数需要返回一个迭代器的时候,逐个返回值。
function number10()
{
for($i = 1; $i <= 10; $i += 1) {
yield $i;
}
}
$generatorObj = number10(); // 是一个对象
// var_dump($generatorObj);
foreach ($generatorObj as $i) {
echo $i;
}
// 输出:12345678910
这和java中的finally一样,经典的try … catch … finally 三段式异常处理。
不论是否捕获到异常,finally中的代码都会执行。只要finally中有return语句,就以finally的返回值为准;否则,以try或者catch中的返回值为准。
function foo() {
try {
echo 'success!'.PHP_EOL;
throw new Exception('exception!');
echo 'failed!'.PHP_EOL;
return 1;
} catch (Exception $e) {
echo $e->getMessage().PHP_EOL;
return 2;
} finally {
echo "finally!".PHP_EOL;
return 3;
}
}
echo foo();
// 输出:success! exception! finally! 3
对二维数组进行遍历,之前可能需要使用两个foreach,现在只需要使用foreach + list了,但是需确保个数一致。
$arr = [
[1, 2, 3],
[4, 5, 6]
];
foreach ($arr as list($a, $b, $c)) {
echo $a.$b.$c.'
';
}
/*
输出:
123
456
*/
function always_false() {
return false;
}
if (empty(always_false())) {
echo 'This will be printed.';
}
echo array(1, 2, 3)[0];
echo [1, 2, 3][0];
echo "foobar"[0];
// 输出:11f
缺点是缺乏互操作性,在需要和其他语言对接时会比较麻烦。
$passwd = '123456';
// 加密,将原密码生成一个hash散列值
$hashKey = password_hash($passwd, PASSWORD_DEFAULT); // 每次刷新,生成的hash散列值都不一样
echo $hashKey,'
';
//输出结果类似于:$2y$10$Z215AkvS1sFVi4syS9no7eGXCa9mBwiH1BviaQzy4Dh4RvL2/JtpK
// 验证,利用hash散列值对原密码进行验证
if(password_verify($passwd, $hashKey)) {
echo "密码正确!";
} else {
echo "密码错误!";
}
PHP已经实现了strval、intval和floatval的函数来进行强制类型转换。为了达到一致性,PHP5.5新增了boolval函数。
可用来返回二维数组中指定的列。
$records = array(
array('id' => 2135,'name' => 'John'),
array('id' => 3245,'name' => 'Smith'),
array('id' => 5342,'name' => 'Peter')
);
//从结果集中取出 name 列
$names = array_column($records, 'name');
print_r($names);
//从结果集中取出 name 列,并用相应的 id 作为键
$names = array_column($records, 'name', 'id');
print_r($names);
在PHP5.6之前,关键字const定义常量时,只能使用固定的值,且值的类型只能是标量。
在PHP5.6中,对const常量进行了增强,允许常量计算,允许使用包含数字、字符串字面值和常量的表达式结果来定义const常量。常量的值也可以为一个数组,但不能是变量。
const A = 2;
const B = A + 1;
const C = "Hello"." Wolrd!";
const D = array(1, 2, 3);
define('E', 100);
var_dump(A, B, C ,D, E);
// 输出:int(2) int(3) string(12) "Hello Wolrd!" array(3) { [0]=> int(1) [1]=> int(2) [2]=> int(3) } int(100)
现在可以不依赖 func_get_args(), 使用 … 运算符来实现变长参数函数。
function test(...$args)
{
var_dump($args);
}
test(1,2,3);
// 输出:array(3) { [0]=> int(1) [1]=> int(2) [2]=> int(3) }
加入右连接运算符 * 来进行幂运算。 同时还支持简写的 *= 运算符,表示进行幂运算并赋值。
printf(2 ** 3); // 8
$a = 2;
$a **= 3;
printf($a); // 8
use 运算符可以导入外部(其他命名空间)的函数和常量,对应的结构为 use function 和 use const。
namespace Name\Space {
const FOO = 42;
function f() { return __FUNCTION__; }
}
namespace {
use const Name\Space\FOO;
use function Name\Space\f;
echo '常量:'.FOO;
echo '
';
echo '函数:'.f();
}
/*
输出:
常量:42
函数:Name\Space\f
*/
在调用函数的时候,通过 … 操作符可以把数组或者可遍历对象解包到参数列表,这和Ruby等语言中的扩张(splat)操作符类似。
function add($a, $b, $c) {
return $a + $b + $c;
}
$arr = [2, 3];
echo add(1, ...$arr); // 输出:6
可以上传超过2G的大文件。
php://input 是个可以访问请求的原始数据的只读流。 在 PHP 5.6 之前 php://input 打开的数据流只能读取一次; 数据流不支持 seek 操作。 不过,现在依赖于 SAPI 的实现,请求体数据被保存的时候, 它可以打开另一个 php://input 数据流并重新读取。 通常情况下,这种情况只是针对 POST 请求,而不是其他请求方式。