php5.3 PHP5.4 PHP5.5 新特性

PS:不能不知的 php5.3 PHP5.4 PHP5.5 新特性?

  1. PHP 5.3中的新特性

    1. 支持命名空间 (Namespace)
    2. 支持延迟静态绑定(Late Static Binding)
    3. 支持goto语句
    4. 支持闭包、Lambda/Anonymous函数
    5. 新增两个魔术方法__callStatic()和__invoke()
    6. 新增Nowdoc语法
    7. 在类外也可使用const来定义常量
    8. 三元运算符增加了一个快捷书写方式:
    9. HTTP状态码在200-399范围内均被认为访问成功
    10.支持动态调用静态方法

    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 正则表达式函数

    1. 支持命名空间(Namespace)

      毫无疑问,命名空间是PHP5.3所带来的最重要的新特性。有了命名空间的概念,在开发大型站点时,就比较容易设计出灵活的结构,同时避免不同包中的类名或变量名产生冲突。
      在PHP5.3之前,惯例的划分Package的办法是通过目录名来分隔代码文件,代码中的类名则用下划线_来表示目录。例如

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

      这样的命名方式被PEAR、Zend Framework及各种PHP项目广泛采用。虽然该方法可以避免不同包或类库中的类名产生冲突,但在书写代码的时候显得较为麻烦和笨拙。
      在PHP5.3中,则只需要指定不同的命名空间即可,命名空间的分隔符为反斜杆\。
      select.php

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

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

                      <?php
                      namespace Zend\Db;
                      include('select.php');
                      $s = new Zend\Db\Table\Select();
                      $s->test();
                      ?>
                      
    2. 支持延迟静态绑定(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

    3. 支持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';
                      ?>
                      
    4. 支持闭包、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);
                      ?>
                      
    5. 新增两个魔术方法__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
                      ?>
                      
    6. 新增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 <<
      

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

                      <?php
                      // 静态变量
                      function foo()
                      {
                          static $bar = <<
    7. 在类外也可使用const来定义常量

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

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

      并且新增了一种常量定义方式:

                      <?php
                      const CONSTANT = 'Hello World';
                      ?>
                      
    8. 三元运算符增加了一个快捷书写方式

                      ?:
                      

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

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

    9. HTTP状态码在200-399范围内均被认为访问成功

    10. 支持动态调用静态方法

                      <?php
                      class Test{
                          public static function testgo()
                          {
                               echo "gogo!";
                          }
                      }
                      $class = 'Test';
                      $action = 'testgo';
                      $class::$action();  //输出 "gogo!"
                      ?>
                      
    11. 支持嵌套处理异常(Exception)

    12. 新的垃圾收集器(GC),并默认启用

  2. PHP 5.4中的新特性

    1. Buid-in web server

      PHP5.4内置了一个简单的Web服务器,这样在做一些简单程序就方便多了,省去了环境配置的工作,特别对于初学者来说。

                      #把当前目录作为Root Document只需要这条命令即可:
                      php -S localhost:3300
                      #也可以指定其它路径:
                      php -S localhost:3300 -t /path/to/root
                      #还可以指定路由:
                      php -S localhost:3300 router.php
                      

      参考:PHP: Build-in web server

    2. Traits

      Traits提供了一种灵活的代码重用机制,即不像interface一样只能定义方法但不能实现,又不能像class一样只能单继承。至于在实践中怎样使用,还需要深入思考

                      trait SayWorld {
                          public function sayHello() {
                                  parent::sayHello();
                                  echo "World!\n";
                                  echo 'ID:' . $this->id . "\n";
                          }
                      }
      
                      class Base {
                          public function sayHello() {
                                  echo 'Hello ';
                          }
                      }
      
                      class MyHelloWorld extends Base {
                          private $id;
      
                          public function __construct() {
                                  $this->id = 123456;
                          }
      
                          use SayWorld;
                      }
      
                      $o = new MyHelloWorld();
                      $o->sayHello();
      
                      /*will output:
                      Hello World!
                      ID:123456
                      */
                      

      参考:traits

    3. Short array syntax

      PHP5.4提供了数组简短语法:

                      $arr = [1,'james', 'james@fwso.cn'];
                      
    4. Array dereferencing

                      function myfunc() {
                          return array(1,'james', 'james@fwso.cn');
                      }
                      

      我认为比数组简短语法更方便的是dereferencing,以前我们需要这样:

                      $arr = myfunc();
                      echo $arr[1];
                      

      在PHP5.4中这样就行了:

                      echo myfunc()[1];
                      
    5. Upload progress

      Session提供了上传进度支持,通过$_SESSION["upload_progress_name"]就可以获得当前文件上传的进度信息,结合Ajax就能很容易实现上传进度条了。
      参考:上传进度支持(Upload progress in sessions)

    6. JsonSerializable Interface

      实现了JsonSerializable接口的类的实例在json_encode序列化的之前会调用jsonSerialize方法,而不是直接序列化对象的属性。
      参考:JsonSerializable接口

    7. Use mysqlnd by default

      现在mysql, mysqli, pdo_mysql默认使用mysqlnd本地库,在PHP5.4以前需要:

                      $./configure --with-mysqli=mysqlnd
                      

      现在:

                      $./configure --with-mysqli
                      
    8. 实例化类

                      class test{
                          function show(){
                          return 'test';
                          }
                      }
                      echo (new test())->show();
                      
    9. 支持 Class::{expr}() 语法

                      foreach ([new Human("Gonzalo"), new Human("Peter")] as $human) {
                          echo $human->{'hello'}();
                      }
                      
    10. Callable typehint

                      function foo(callable $callback) {
                      }
                        则:
                        foo("false"); //错误,因为false不是callable类型
                        foo("printf"); //正确
                        foo(function(){}); //正确
                      class A {
                        static function show() {}
                      }
                      foo(array("A", "show")); //正确
                      
    11. 函数类型提示的增强

                      由于php是弱类型的语言,因此在php 5.0后,引入了函数类型提示的功能,其含义为对于传入函数中的参数都进行类型检查,举个例子,有如下的类:
                      class bar {
                      function foo(bar $foo) {
                      }
                      //其中函数foo中的参数规定了传入的参数必须为bar类的实例,否则系统会判断出错。同样对于数组来说,也可以进行判断,比如:
                      function foo(array $foo) {
                      }
                      }
                      foo(array(1, 2, 3)); // 正确,因为传入的是数组
                      foo(123); // 不正确,传入的不是数组
                      
    12. 新增加了$_SERVER["REQUEST_TIME_FLOAT"],这个是用来统计服务请求时间的,并用ms来表示

                      echo "脚本执行时间 ", round(microtime(true) - $_SERVER["REQUEST_TIME_FLOAT"], 2), "s";
                      
    13. 让Json更懂中文(JSON_UNESCAPED_UNICODE)

                      echo json_encode("中文", JSON_UNESCAPED_UNICODE);
                      //"中文"
                      
    14. 二进制直接量(binary number format)

                      $bin  = 0b1101;
                      echo $bin;
                      //13
                      

    PHP5.4中其它值得注意的改变

            PHP 5.4.0 性能大幅提升, 修复超过100个bug.
            废除了register_globals, magic_quotes以及安全模式。
            另外值得一提的是多字节支持已经默认启用了,
            default_charset从ISO-8859-1已经变为UTF-8.
            默认发送“Content-Type: text/html; charset=utf-8”,
            你再也不需要在HTML里写meta tag,也无需为UTF-8兼容而传送额外的header了。
    
            删除的特性
            最后,我们集中整理了几年来标记为已弃用的多个特性。这些特性包括 allow_call_time_pass_reference、define_syslog_variables、highlight.bg、register_globals、register_long_arrays、magic_quotes、safe_mode、zend.ze1_compatibility_mode、session.bug_compat42、session.bug_compat_warn 以及 y2k_compliance。
            除了这些特性之外,magic_quotes 可能是最大的危险。在早期版本中,未考虑因 magic_quotes 出错导致的后果,简单编写且未采取任何举措使自身免受 SQL 注入攻击的应用程序都通过 magic_quotes 来保护。如果在升级到 PHP 5.4 时未验证已采取正确的 SQLi 保护措施,则可能导致安全漏洞。
    
            其他改动和特性
            有一种新的“可调用的”类型提示,用于某方法采用回调作为参数的情况。
            htmlspecialchars() 和 htmlentities() 现在可更好地支持亚洲字符,如果未在 php.ini 文件中显式设置 PHP default_charset,这两个函数默认使用 UTF-8 而不是 ISO-8859-1。
            
    
  3. PHP 5.5中的新特性

                新特性及提案列表 都相当大,而且不是按重要性排序。所以,如果你不想通读一遍的话,这里有四个特点我个人是最兴奋的:
                :一个简单的密码散列API
                :标量类型提示
                :Getter和Setter
                :生成器
                现在,让我们来看看PHP5.5 可能会新增的功能:
                1、放弃对Windows XP和2003 的支持
    
                2.弃用e修饰符
                e修饰符是指示preg_replace函数用来评估替换字符串作为PHP代码,而不只是仅仅做一个简单的字符串替换。不出所料,这种行为会源源不断的出现安全问题。这就是为什么在PHP5.5 中使用这个修饰符将抛出一个弃用警告。作为替代,你应该使用preg_replace_callback函数。你可以从RFC找到更多关于这个变化相应的信息。
    
                3.新增函数和类
                boolval()
                PHP已经实现了strval、intval和floatval的函数。为了达到一致性将添加boolval函数。它完全可以作为一个布尔值计算,也可以作为一个回调函数。
    
                hash_pbkdf2()
                PBKDF2全称“Password-Based Key Derivation Function 2”,正如它的名字一样,是一种从密码派生出加密密钥的算法。这就需要加密算法,也可以用于对密码哈希。更广泛的说明和用法示例
    
                array_column()
                $userNames = array_column($users, 'name');
                // is the same as
                $userNames = [];
                foreach ($users as $user) {
                    $userNames[] = $user['name'];
                }
    
                intl 扩展
                将有许多改进 intl的扩展。例如,将会有新的IntlCalendar,IntlGregorianCalendar,IntlTimeZone,IntlBreakIterator,IntlRuleBasedBreakIterator,IntlCodePointBreakIterator类。之前,我竟然不知道有这么多关于intl扩展,如果你想知道更多,我建议你去最新公告里找 Calendar和 BreakIterator。
    
                4.一个简单的密码散列API
                $password = "foo";
                // creating the hash
                $hash = password_hash($password, PASSWORD_BCRYPT);
                // verifying a password
                if (password_verify($password, $hash)) {
                    // password correct!
                } else {
                    // password wrong!
                }
    
                5.新的语言特性和增强功能。
                常量引用
                “常量引用”意味着数组可以直接操作字符串和数组字面值。举两个例子:
                function randomHexString($length) {
                    $str = '';
                    for ($i = 0; $i < $length; ++$i) {
                        $str .= "0123456789abcdef"[mt_rand(0, 15)]; // direct dereference of string
                    }
                }
                function randomBool() {
                    return [false, true][mt_rand(0, 1)]; // direct dereference of array
                }
                我不认为在实践中会使用此功能,但它使语言更加一致。请参阅 RFC。
    
                6.调用empty()函数(和其他表达式)一起工作
                目前,empty()语言构造只能用在变量,而不能在其他表达式。
                在特定的代码像empty($this->getFriends())将会抛出一个错误。作为PHP5.5 这将成为有效的代码
    
                7.获取完整类别名称
                PHP5.3 中引入命名空间的别名类和命名空间短版本的功能。虽然这并不适用于字符串类名称
                use Some\Deeply\Nested\Namespace\FooBar;
                // does not work, because this will try to use the global `FooBar` class
                $reflection = new ReflectionClass('FooBar');
                echo FooBar::class;
                为了解决这个问题采用新的FooBar::class语法,它返回类的完整类别名称
    
                8.参数跳跃
                如果你有一个函数接受多个可选的参数,目前没有办法只改变最后一个参数,而让其他所有参数为默认值。
                RFC上的例子,如果你有一个函数如下:
                function create_query($where, $order_by, $join_type='', $execute = false, $report_errors = true) { ... }
                那么有没有办法设置$report_errors=false,而其他两个为默认值。为了解决这个跳跃参数的问题而提出:
                create_query("deleted=0", "name", default, default, false);
                我个人不是特别喜欢这个提议。在我的眼睛里,代码需要这个功能,只是设计不当。函数不应该有12个可选参数。
    
                9.标量类型提示
                标量类型提示原本计划进入5.4,但由于缺乏共识而没有做。获取更多关于为什么标量类型提示没有做进PHP的信息,请参阅: 标量类型提示比你认为的更难。
                对于PHP5.5 而言,针对标量类型提示讨论又一次出现,我认为这是一个相当不错的 提议。
                它需要通过输入值来指定类型。例如:123,123.0,“123”都是一个有效的int参数输入,但“hello world”就不是。这与内部函数的行为一致。
                function foo(int $i) { ... }
                foo(1);      // $i = 1
                foo(1.0);    // $i = 1
                foo("1");    // $i = 1
                foo("1abc"); // not yet clear, maybe $i = 1 with notice
                foo(1.5);    // not yet clear, maybe $i = 1 with notice
                foo([]);     // error
                foo("abc");  // error
    
                10.Getter 和 Setter
                如果你从不喜欢写这些getXYZ()和setXYZ($value)方法,那么这应该是你最受欢迎的改变。提议添加一个新的语法来定义一个属性的设置/读取:
                <?php
    
                class TimePeriod {
                    public $seconds;
    
                    public $hours {
                        get { return $this->seconds / 3600; }
                        set { $this->seconds = $value * 3600; }
                    }
                }
                $timePeriod = new TimePeriod;
                $timePeriod->hours = 10;
                var_dump($timePeriod->seconds); // int(36000)
                var_dump($timePeriod->hours);   // int(10)
                当然还有更多的功能,比如只读属性。如果你想要知道更多,请参阅 RFC。
    
                11.生成器
                目前,自定义迭代器很少使用,因为它们的实现,需要大量的样板代码。生成器解决这个问题,并提供了一种简单的样板代码来创建迭代器。
                例如,你可以定义一个范围函数作为迭代器:
                <?php
                function *xrange($start, $end, $step = 1) {
                    for ($i = $start; $i < $end; $i += $step) {
                        yield $i;
                    }
                }
                foreach (xrange(10, 20) as $i) {
                    // ...
                }
                上述xrange函数具有与内建函数相同的行为,但有一点区别:不是返回一个数组的所有值,而是返回一个迭代器动态生成的值。
    
                12.列表解析和生成器表达式
    
                列表解析提供一个简单的方法对数组进行小规模操作:
                $firstNames = [foreach ($users as $user) yield $user->firstName];
                上述列表解析相等于下面的代码:
                $firstNames = [];
                foreach ($users as $user) {
                    $firstNames[] = $user->firstName;
                }
                也可以这样过滤数组:
                $underageUsers = [foreach ($users as $user) if ($user->age < 18) yield $user];
                生成器表达式也很类似,但是返回一个迭代器(用于动态生成值)而不是一个数组。
    
                以上来自:http://www.oschina.net/question/157182_61259
    
                13.finally关键字
                这个和java中的finally一样,经典的try ... catch ... finally 三段式异常处理。
    
                14.foreach 支持list()
                对于“数组的数组”进行迭代,之前需要使用两个foreach,现在只需要使用foreach + list了,但是这个数组的数组中的每个数组的个数需要一样。看文档的例子一看就明白了。
                $array = [
                    [1, 2],
                    [3, 4],
                ];
                foreach ($array as list($a, $b)) {
                    echo "A: $a; B: $b\n";
                }
    
                15.增加了opcache扩展
                使用opcache会提高php的性能,你可以和其他扩展一样静态编译(--enable-opcache)或者动态扩展(zend_extension)加入这个优化项。
    
                16.非变量array和string也能支持下标获取了
                echo array(1, 2, 3)[0];
                echo [1, 2, 3][0];
                echo "foobar"[2];
            

总结:
1:总结内容
2:总结内容
3:总结内容


参考:
1.参考内容
2.参考内容

相关文章
  1. $_GET,$_POST与urldecode的使用风险
  2. yii2框架的错误处理
  3. Yii2过滤器-behaviors()行为调用
  4. 密码保护:PHP多进程编程
  5. 全排列算法原理和实现
  6. PHP静态方法和非静态方法的使用场景
本站版权
1、本站所有主题由该文章作者发表,该文章作者与尘埃享有文章相关版权
2、其他单位或个人使用、转载或引用本文时必须同时征得该文章作者和尘埃的同意
3、本帖部分内容转载自其它媒体,但并不代表本站赞同其观点和对其真实性负责
4、如本帖侵犯到任何版权问题,请立即告知本站,本站将及时予与删除并致以最深的歉意
5、原文链接:
二维码
Posted in php, 编程语言

发表评论