技术饭

PHP7新特性

copylian    0 评论    5608 浏览    2022.01.24

PHP7已经更新很久了,但是实际上很多人用的还是php5.6,最近也刚刚开始使用php7,发现还是有很多的新特性,事实上现在已经更新到php8了,php7性能方面比以往的版本好很多,php7新增了一些特性也丢弃掉了一些用法。

1、Generators (PHP 5 >= 5.5.0, PHP 7),参考:PHP 中的生成器(Generator)详解

通过添加 yield 关键字支持了 generators,Generators 提供了一个更简单的方法实现迭代器,不需要实现 Iterator 接口。

<?php

    function xrange($start, $limit, $step = 1) {

        for ($i = $start; $i <= $limit; $i += $step) {

            yield $i;

        }

    }

 

    echo 'Single digit odd numbers: ';

 

    /* 注意保存在内存中的数组绝不会被创建或返回 */

    foreach (xrange(1, 9, 2) as $number) {

        echo "$number ";

    }

?>

以上例程会输出:

Single digit odd numbers: 1 3 5 7 9


2、新增 finally 关键字 (PHP 5 >= 5.5.0, PHP 7),PHP中try、catch、finally的执行顺序

try-catch 现在支持 finally

<?php

try {    

    //好好干,出了问题不要怕,外面有人接应

} catch (HttpException $e) {    

    //时刻准备着,处理上面抛出的HTTP问题

} catch (Exception $e) {    

    //时刻准备着,处理他们都处理不了的问题

} finally {    

    //打扫战场,都收拾好了再走人

}

?>

try内部正常执行try的内部逻辑,异常则执行catch的内部逻辑结构,但是不管执行的哪个都会执行完try catch的内部逻辑(非return)后执行finally的内部逻辑。

如果try catch都有return,按照正常执行,然后执行finally的逻辑,再返回对应的try 或者catch里执行return。

如果try catch finally都有return,执行完finally的逻辑后,会调用finally的return。


3、foreach 现在支持 list() (PHP 5 >= 5.5.0, PHP 7)

foreach 控制结构现在支持通过 list() 构造将嵌套数组分离到单独的变量。例如:<?php

$array = [

    [1, 2],

    [3, 4],

];


foreach ($array as list($a, $b)) {

    echo "A: $a; B: $b\n";

}

?>



以上例程会输出:

A: 1; B: 2

A: 3; B: 4


4、array_column (PHP 5 >= 5.5.0, PHP 7)

array_column — 返回数组中指定的一列

<?php

// 表示由数据库返回的可能记录集的数组

$a = array(

    array(

        'id' => 5698,

        'first_name' => 'Bill',

        'last_name' => 'Gates',

    ),

    array(

        'id' => 4767,

        'first_name' => 'Steve',

        'last_name' => 'Jobs',

    )

    array(

        'id' => 3809,

        'first_name' => 'Mark',

        'last_name' => 'Zuckerberg',

    )

);

$last_names = array_column($a, 'last_name', 'id');

print_r($last_names);

?>

输出:

Array(

    [5698] => Gates

    [4767] => Jobs

    [3809] => Zuckerberg

)


5、使用表达式定义常量 (PHP 5 >= 5.6.0, PHP 7)

在之前的 PHP 版本中, 必须使用静态值来定义常量,声明属性以及指定函数参数默认值。 现在你可以使用包括数值、字符串字面量以及其他常量在内的数值表达式来 定义常量、声明属性以及设置函数参数默认值。

<?php

const ONE = 1;

const TWO = ONE * 2;

 

class C {

    const THREE = TWO + 1;

    const ONE_THIRD = ONE / self::THREE;

    const SENTENCE = 'The value of THREE is '.self::THREE;

 

    public function f($a = ONE + self::THREE) {

        return $a;

    }

}

 

echo (new C)->f()."\n";

echo C::SENTENCE;

?>

以上例程会输出:

4

The value of THREE is 3


现在可以通过 const 关键字来定义类型为 array 的常量。

<?php

const ARR = ['a', 'b'];

echo ARR[0];

?>

以上例程会输出:

a


6、使用 ... 运算符定义变长参数函数 (PHP 5 >= 5.6.0, PHP 7)

现在可以不依赖 func_get_args(), 使用 ... 运算符 来实现 变长参数函数。

<?php

function f($req, $opt = null, ...$params) {

    // $params 是一个包含了剩余参数的数组

    printf('$req: %d; $opt: %d; number of params: %d'."\n",

    $req, $opt, count($params));

}


f(1);

f(1, 2);

f(1, 2, 3);

f(1, 2, 3, 4);

f(1, 2, 3, 4, 5);

?>


以上例程会输出:

$req: 1; $opt: 0; number of params: 0

$req: 1; $opt: 2; number of params: 0

$req: 1; $opt: 2; number of params: 1

$req: 1; $opt: 2; number of params: 2

$req: 1; $opt: 2; number of params: 3

使用 ... 运算符进行参数展开 (PHP 5 >= 5.6.0, PHP 7)

在调用函数的时候,使用 ... 运算符, 将 数组 和 可遍历 对象展开为函数参数。 在其他编程语言,比如 Ruby中,这被称为连接运算符,。

<?php

function add($a, $b, $c) {

    return $a + $b + $c;

}

 

$operators = [2, 3];

echo add(1, ...$operators);

?>

  

以上例程会输出:

6


7、use function 以及 use const (PHP 5 >= 5.6.0, PHP 7)

use 运算符 被进行了扩展以支持在类中导入外部的函数和常量。 对应的结构为 use function 和 use const。

<?php

namespace Name\Space {

    const FOO = 42;

    function f() { echo __FUNCTION__."\n"; }

}

 

namespace {

     use const Name\Space\FOO;

     use function Name\Space\f;

     echo FOO."\n";

     f();

}

?>

以上例程会输出:

42

Name\Space\f


8、__debugInfo() (PHP 5 >= 5.6.0, PHP 7)

加入 __debugInfo(), 当使用 var_dump() 输出对象的时候, 可以用来控制要输出的属性和值。<?php

class C {

     private $prop;

 

     public function __construct($val) {

          $this->prop = $val;

     }

 

     public function __debugInfo() {

          return [

               'propSquared' => $this->prop ** 2,

          ];

    }

}

var_dump(new C(42));

?>

以上例程会输出:

object(C)#1 (1) {

 ["propSquared"]=>

 int(1764)

}


9、标量类型声明 (PHP 7)

标量类型声明 有两种模式: 强制 (默认) 和 严格模式。 现在可以使用下列类型参数(无论用强制模式还是严格模式): 字符串(string), 整数 (int), 浮点数 (float), 以及布尔值 (bool)。它们扩充了PHP5中引入的其他类型:类名,接口,数组和 回调类型。

<?php

// Coercive mode

function sumOfInts(int ...$ints)

{

     return array_sum($ints);

}

 

var_dump(sumOfInts(2, '3', 4.1));

以上例程会输出:

int(9)

要使用严格模式,一个 declare 声明指令必须放在文件的顶部。这意味着严格声明标量是基于文件可配的。 这个指令不仅影响参数的类型声明,也影响到函数的返回值声明(参见返回值类型声明, 内置的PHP函数以及扩展中加载的PHP函数)


10、返回值类型声明 (PHP 7)

PHP 7 增加了对返回类型声明的支持。 类似于参数类型声明,返回类型声明指明了函数返回值的类型。可用的类型与参数声明中可用的类型相同。 

<?php

function arraysSum(array ...$arrays): array

{

     return array_map(function(array $array): int {

         return array_sum($array);

     }, $arrays);

}

print_r(arraysSum([1,2,3], [4,5,6], [7,8,9]));


以上例程会输出:

Array

(
            [0] => 6
            [1] => 15
            [2] => 24

)


11、null合并运算符 (PHP 7)

由于日常使用中存在大量同时使用三元表达式和 isset()的情况, 我们添加了null合并运算符 (??) 这个语法糖。如果变量存在且值不为NULL, 它就会返回自身的值,否则返回它的第二个操作数。

<?php

// Fetches the value of $_GET['user'] and returns 'nobody'

// if it does not exist.

$username = $_GET['user'] ?? 'nobody';

// This is equivalent to:

$username = isset($_GET['user']) ? $_GET['user'] : 'nobody';

 

// Coalesces can be chained: this will return the first

// defined value out of $_GET['user'], $_POST['user'], and

// 'nobody'.

$username = $_GET['user'] ?? $_POST['user'] ?? 'nobody';

?>

 

12、太空船操作符(组合比较符)(PHP 7)

太空船操作符用于比较两个表达式。当$a小于、等于或大于$b时它分别返回-1、0或1。 比较的原则是沿用 PHP 的常规比较规则进行的。

<?php

// Integers

echo 1 <=> 1; // 0

echo 1 <=> 2; // -1

echo 2 <=> 1; // 1

 

// Floats

echo 1.5 <=> 1.5; // 0

echo 1.5 <=> 2.5; // -1

echo 2.5 <=> 1.5; // 1

  

// Strings

echo "a" <=> "a"; // 0

echo "a" <=> "b"; // -1

echo "b" <=> "a"; // 1

?>


13、通过 define() 定义常量数组 (PHP 7)

Array 类型的常量现在可以通过 define() 来定义。在 PHP5.6 中仅能通过 const 定义。

<?php

define('ANIMALS', [

     'dog',

     'cat',

     'bird'

]);

echo ANIMALS[1]; // outputs "cat"

?>


14、匿名类 (PHP 7)

现在支持通过new class 来实例化一个匿名类,这可以用来替代一些“用后即焚”的完整类定义。

<?php

interface Logger {

     public function log(string $msg);

}

 

class Application {

     private $logger;

 

     public function getLogger(): Logger {

           return $this->logger;

     }

 

     public function setLogger(Logger $logger) {

           $this->logger = $logger;

     }

    }

 

    $app = new Application;

    $app->setLogger(new class implements Logger {

         public function log(string $msg) {

              echo $msg;

         }

    });

 

    var_dump($app->getLogger());

?>

以上例程会输出:

object(class@anonymous)#2 (0) {}


15、Closure::call() (PHP 7)

Closure::call() 现在有着更好的性能,简短干练的暂时绑定一个方法到对象上闭包并调用它。

<?php

class A {private $x = 1;}

 

// Pre PHP 7 code

$getXCB = function() {return $this->x;};

$getX = $getXCB->bindTo(new A, 'A'); // intermediate closure

echo $getX();

 

// PHP 7+ code

$getX = function() {return $this->x;};

echo $getX->call(new A);

以上例程会输出:

1

1


16、为unserialize()提供过滤 (PHP 7)

这个特性旨在提供更安全的方式解包不可靠的数据。它通过白名单的方式来防止潜在的代码注入。

<?php

// converts all objects into __PHP_Incomplete_Class object

$data = unserialize($foo, ["allowed_classes" => false]);

 

// converts all objects into __PHP_Incomplete_Class object except those of MyClass and MyClass2

$data = unserialize($foo, ["allowed_classes" => ["MyClass", "MyClass2"]);

 

// default behaviour (same as omitting the second argument) that accepts all classes

$data = unserialize($foo, ["allowed_classes" => true]);


17、Group use declarations (PHP 7)

从同一 namespace 导入的类、函数和常量现在可以通过单个 use 语句 一次性导入了。

<?php

// Pre PHP 7 code

use some\namespace\ClassA;

use some\namespace\ClassB;

use some\namespace\ClassC as C;

 

use function some\namespace\fn_a;

use function some\namespace\fn_b;

use function some\namespace\fn_c;

 

use const some\namespace\ConstA;

use const some\namespace\ConstB;

use const some\namespace\ConstC;

 

// PHP 7+ code

use some\namespace\{ClassA, ClassB, ClassC as C};

use function some\namespace\{fn_a, fn_b, fn_c};

use const some\namespace\{ConstA, ConstB, ConstC};

?>


18、新增了整除函数 intdiv(),使用实例:

<?php

    var_dump(intdiv(10, 3));

?>

以上实例会输出:

int(3)


19、PHP 7 Session 选项

PHP 7 session_start() 函数可以接收一个数组作为参数,可以覆盖 php.ini 中 session 的配置项。

这个特性也引入了一个新的 php.ini 设置(session.lazy_write), 默认情况下设置为 true,意味着 session 数据只在发生变化时才写入。

除了常规的会话配置指示项, 还可以在此数组中包含 read_and_close 选项。如果将此选项的值设置为 TRUE, 那么会话文件会在读取完毕之后马上关闭, 因此,可以在会话数据没有变动的时候,避免不必要的文件锁。把 cache_limiter 设置为私有的,同时在阅读完 session 后立即关闭。

实例

<?php

session_start([

    'cache_limiter' => 'private',

    'read_and_close' => true

]);

?>

session_start([

    'cache_limiter' => 'private', //在读取完毕会话数据之后马上关闭会话存储文件

    'cookie_lifetime'=>3600,   //SessionID在客户端Cookie储存的时间,默认是0,代表浏览器一关闭SessionID就作废

    'read_and_close'=>true   //在读取完会话数据之后, 立即关闭会话存储文件,不做任何修改

]);

$_SESSION['name']='quan';

echo $_SESSION['name'];


20、PHP CSPRNG:random_bytes()、random_int()

CSPRNG(Cryptographically Secure Pseudo-Random Number Generator,伪随机数产生器)。PHP 7 通过引入几个 CSPRNG 函数提供一种简单的机制来生成密码学上强壮的随机数。

random_bytes() - 加密生存被保护的伪随机字符串。

random_int() - 加密生存被保护的伪随机整数。

<?php

    $bytes = random_bytes(5);

    print(bin2hex($bytes));

?>

以上程序执行输出结果为:6f36d48a29

<?php

    print(random_int(100, 999));

    print(PHP_EOL);

    print(random_int(-1000, 0));

?>

以上程序执行输出结果为:

723

-64


参考资料:

PHP7新特性总结

只袄早~~~
感谢你的支持,我会继续努力!
扫码打赏,感谢您的支持!
PHP7 新特性 

文明上网理性发言!

  • 还没有评论,沙发等你来抢