cncml手绘网

标题: 【PHP系列】PHP 7.0新增特性详解 [打印本页]

作者: admin    时间: 2018-6-21 22:38
标题: 【PHP系列】PHP 7.0新增特性详解
开始介绍PHP7.0新特性,具体的可以参照官网的介绍,我来挑一些给大家详细讲解下
http://php.net/manual/en/migration70.new-features.php
8 m" k/ D# G! F, ?, h0 m9 `: Q1. ?? 运算符(NULL 合并运算符)
5 t3 t& c! I: v. g- T
  1. $a = $_GET['a'] ?? 1;
复制代码

( `4 o/ t3 s" c( x5 H
它相当于:
  1. $a = empty($_GET['a']) ? 1 : $_GET['a'];
复制代码

& \4 A- J2 u4 \) q) p7 h' v7 V
我们知道三元运算符是可以这样用的:
  1. $a ?: 1
复制代码
* p! e7 K/ R1 `
但是这是建立在 $a 已经定义了的前提上。新增的 ?? 运算符可以简化判断。简化代码的同时也更直观了!
/ y% U' z* ]) G4 ?9 J
2. 函数返回值类型声明
官方文档提供的例子(注意 ... 的边长参数语法在 PHP 5.6 以上的版本中才有):
* C$ f1 w5 u5 U- O' ~8 Z% S
  1. <?php2 ~' I6 p: D2 [* E' m
  2. function arraysSum(array ...$arrays): array
      x: l; g) m: s
  3. {. k. [8 C; T1 O5 U* s& y: }0 R
  4.     return array_map(function(array $array): int {
    - R2 s* I7 [% n( B. e0 s" V% Z' d
  5.          return array_sum($array);
    0 C: F9 K# x# d  {5 z
  6.     }, $arrays);5 h! U2 l7 X6 s* {) p. q
  7. }
    0 n+ K, V# I$ D7 d9 K# y( d

  8. % j0 `$ y! Z) H; B7 K
  9. print_r(arraysSum([1,2,3], [4,5,6], [7,8,9]));
复制代码
" ~; o0 `' x/ }

# l; A; Q& T2 ~% k
从这个例子中可以看出现在函数(包括匿名函数)都可以指定返回值的类型。
这个特性可以帮助我们避免一些 PHP 的隐式类型转换带来的问题。在定义一个函数之前就想好预期的结果可以避免一些不必要的错误。
不过这里也有一个特点需要注意。PHP 7 增加了一个 declare 指令:strict_types,既使用严格模式。
使用返回值类型声明时,如果没有声明为严格模式,如果返回值不是预期的类型,PHP 还是会对其进行强制类型转换。但是如果是严格模式, 则会出发一个 TypeError 的 Fatal error。
强制模式:
  1. <?phpfunction foo($a) : int% t; W) D  ~" ?  @; j5 ]& q
  2. {
    * m# Z7 h( G% `$ j
  3.     return $a;) l) w9 t* T& {6 q/ t
  4. }
    4 h9 \# ^  A3 s8 k9 s

  5. ( C' Q* c7 F/ p. B3 E; o
  6. foo(1.0);
复制代码
/ w0 J9 X3 R  s
' O0 J3 F* h; M4 ?  h
以上代码可以正常执行,foo 函数返回 int 1,没有任何错误。
严格模式:
  1. <?php* z' P, V; D, \9 A+ U* {
  2. declare(strict_types=1);
    9 f% Q6 d$ D. ]+ W* V- m1 T

  3. ' I8 v' x+ G* W' l- i) p3 g
  4. function foo($a) : int' }+ x5 w0 H$ z6 _
  5. {: X3 I5 y( l+ o
  6.     return $a;
    ' G' ~& w$ G" |0 s( ?; X
  7. }
    6 `8 a* p; h9 j0 y6 c
  8. , t. s4 r  Y9 m1 x- ~$ p
  9. foo(1.0);
复制代码
; \9 O( M% A5 ?
% ]. C2 f. R& b2 V3 f
在声明之后,就会触发致命错误。
# PHP Fatal error:  Uncaught TypeError: Return value of foo() must be of the type integer, float returned in test.php:66 k  K+ n# a3 T$ U$ C
3. 标量类型声明
PHP 7 中的函数的形参类型声明可以是标量了。在 PHP 5 中只能是类名、接口、array 或者 callable (PHP 5.4,即可以是函数,包括匿名函数),现在也可以使用 string、int、float和 bool 了。
  1. <?php
    0 _2 E5 O' R/ R  C$ @; h
  2. // Coercive mode1 J& F) P& x( d; Q5 O2 c7 Q0 h3 [
  3. function sumOfInts(int ...$ints)' e2 r/ r8 O& w/ q
  4. {
    : a8 X5 E+ j2 H# h
  5.     return array_sum($ints);
    * I# {! \- z: j' T0 o/ v) z
  6. }( M/ c+ u( f- J1 o
  7. 8 k% D/ x) Q! H* F7 R
  8. var_dump(sumOfInts(2, '3', 4.1));
复制代码

; E5 q' ^# K. i5 c% ~& b. ^
1 n5 B$ T5 R2 B" m* `1 X
需要注意的是上文提到的严格模式的问题在这里同样适用:强制模式(默认,既强制类型转换)下还是会对不符合预期的参数进行强制类型转换,严格模式下则触发 TypeError 的致命错误。
4. use 批量声明
PHP 7 中 use 可以在一句话中声明多个类或函数或 const 了:
  1. <?php4 f/ P. F) v( }0 O( D! p5 e2 B
  2. use some\namespace\{ClassA, ClassB, ClassC as C};
    , u$ {/ V' }; @6 S0 q- C) l( _
  3. use function some\namespace\{fn_a, fn_b, fn_c};7 n. ?/ C) n! @9 H% L: I
  4. use const some\namespace\{ConstA, ConstB, ConstC};
复制代码

$ {  A5 d( m; j: \1 C& j
但还是要写出每个类或函数或 const 的名称(并没有像 python 一样的 from some import * 的方法)。
需要留意的问题是:如果你使用的是基于 composer 和 PSR-4 的框架,这种写法是否能成功的加载类文件?其实是可以的,composer 注册的自动加载方法是在类被调用的时候根据类的命名空间去查找位置,这种写法对其没有影响。
再来简单说几个:
1、PHP 5.3 开始有了匿名函数,现在又有了匿名类了;
2、define 现在可以定义常量数组;
3、闭包( Closure)增加了一个 call 方法;
4、生成器(或者叫迭代器更合适)可以有一个最终返回值(return),也可以通过 yield from 的新语法进入一个另外一个生成器中(生成器委托)。
生成器的两个新特性(return 和 yield from)可以组合。具体的大家可以自行测试。

6 ?2 L5 n2 z. j# k" g
. F' n, a9 M; t/ z




欢迎光临 cncml手绘网 (http://bbs.cncml.com/) Powered by Discuz! X3.2