php7.4新特性

短闭包,可实现更简洁的单行书写
预加载,以提高性能
类中的类型属性
自定义对象序列化添加一种(反)序列化对象的新方法
改进类型差异
简化 Null 合并运算符
FFI 为 PHP 扩展开发开辟新机遇
弃用短开标签
在数组中支持展开运算符
阅读以下内容了解更多
短闭包
短闭包实现更简洁的单行书写。

array_map(function (User $user) { 

    return $user->id; 

}, $users)


array_map(fn(User $user) => $user->id, $users)

关于短闭包的一些注意事项:

可以访问父作用域,不需要 use 关键字。
$this 可以像普通的闭包一样使用。
短闭包只能包含一行,也就是 return 语句。
你可以在这里深入阅读它们。

类属性可以提示类型:

class A

{

    public string $name;



    public Foo $foo;

}

我之前写过 PHP 的类型体系,所以很高兴看到一些 PHP 的核心被改进了。

类型差异是一个值得博客文章讨论的主题;简而言之:你将能够使用协变量返回类型。。。

class ParentType {}

class ChildType extends ParentType {}



class A

{

    public function covariantReturnTypes(): ParentType

    { /* … */ }

}



class B extends A

{

    public function covariantReturnTypes(): ChildType

    { /* … */ }

}
class A

{

    public function contraVariantArguments(ChildType $type)

    { /* … */ }

}



class B extends A

{

    public function contraVariantArguments(ParentType $type)

    { /* … */ }

}

不再需要这样做:

$data['date'] = $data['date'] ?? new DateTime();

你可以这样做:

$data['date'] ??= new DateTime();

现在你可以在数组中使用展开运算符:

$arrayA = [1, 2, 3];

$arrayB = [4, 5];

$result = [0, ...$arrayA, ...$arrayB, 6 ,7];

// [0, 1, 2, 3, 4, 5, 6, 7]

请注意,这仅适用于带数字键的数组。

外部函数接口
外部函数接口,简称 FFI,允许从用户区调用 C 代码。这意味着 PHP 扩展可以用纯 PHP 编写。

应该指出,这是一个复杂的话题。您仍然需要 C 知识才能正确使用此功能。

预加载是 PHP 核心的一个振奋人心的新功能,可以带来不可预估的性能改进。

简而言之:如果您今天使用的是框架,则必须在每次请求时加载和重新编译其文件。预加载允许服务器在启动时于内存中加载 PHP 文件,并使它们持久化可用于所有后续请求(只要不停电)。

性能提升当然需要付出代价:如果预加载文件的源文件发生变化,则必须重新启动服务器(该部分若有异议,请查看 详细介绍)

增加了两个新的魔术方法:__serialize 和 __unserialize 。这些方法和 __sleep 以及 __wakeup 的不同之处已经在 中进行了讨论。

如果你书写了类似以下的内容:

echo "sum: " . $a + $b;

PHP 之前会像这样编译它:

echo ("sum: " . $a) + $b;

而 PHP 8 将使它如下编译:

echo "sum :" . ($a + $b);

当在遇到没有圆括号包含的 ‘+’ 或 ’ - ’ 表达式之前有 ‘.’ 的时候,PHP 7.4 会提示弃用警告。

这在技术上不是与 PHP 7.4 相关的更新,但值得一提的是:的投票规则已经改变。

他们将总是需要 2/3 的大半支持才能通过。
投票时间不短,所有必须至少开放 2 周。
参考文献的反思
像 Symfony 的 var dumper 这样的库严重依赖于反射 API 来可靠地转储变量。以前,没有对参考文献提供适当的反映支持,导致这些库依赖于黑客来检测反映。

PHP 7. 4 添加了 ReflectionReference 解决此问题的类。

添加 mb_str_split 函数
此函数提供与 str_split 多字节字符串相同的功能。

永久支持 ext-hash
正如标题所说,此扩展现在可在所有 PHP 安装中永久支持使用。

默认不启用 PEAR EXTERNALS
由于 PEAR 不再支持维护,核心团队决定在 PHP 7.4 中删除它的默认安装。

密码哈希注册表
对如何使用散列库进行内部更改,以便用户可以更轻松地使用它们。

弃用ext/wwdx
此数据交换格式从未标准化,现在已经弃用该扩展。

PHP 短标签被弃用
短开标签 <? 已被弃用,将在 PHP 8 中删除。短声明标记 <?= 不受影响。

左关联三元运算符被弃用
三元运算符在 PHP 中有一些奇怪的怪癖。此 为嵌套的三元语句添加了弃用。在 PHP 8 中,此弃用将转换为编译时错误。

1 ? 2 : 3 ? 4 : 5;   // deprecated

(1 ? 2 : 3) ? 4 : 5; // ok

向后不兼容的更改 UPGRADING
升级PHP版本时,您应该始终查看完整的 UPGRADING 文档。

以下是一些突出显示的向后不兼容的更改:

调用 var_dump 一个 DateTime 或 DateTimeImmutable 实例后面将不再保留对象的可访问属性。
openssl_random_pseudo_bytes 将在错误情况下抛出异常。
尝试序列化一个 PDO 或 PDOStatement 实例将生成一个 Exception 而不是一个 PDOException 。
调用 get_object_vars() 上的 ArrayObject 实例将返回的属性 ArrayObject 本身,而不是包装的数组或对象的值。请注意,(array) 强制转换不受影响。

1.开启Opcache

zend_extension=opcache.so 
opcache.enable=1 
opcache.enable_cli=1

2.使用GCC 4.8以上进行编译
只有GCC 4.8以上PHP才会开启Global Register for opline and execute_data支持, 这个会带来5%左右的性能提升(Wordpres的QPS角度衡量)

3.开启HugePage (根据系统内存决定)

4.PGO (Profile Guided Optimization)
第一次编译成功后,用项目代码去训练PHP,会产生一些profile信息,最后根据这些信息第二次gcc编译PHP就可以得到量身定做的PHP7

需要选择在你要优化的场景中: 访问量最大的, 耗时最多的, 资源消耗最重的一个页面.

参考: http://www.laruence.com/2015/06/19/3063.html
参考: http://www.laruence.com/2015/12/04/3086.html

如何更好的写代码来迎接PHP7?
不使用php7废弃的方法,扩展
使用2个版本都兼容的语法特性【 list ,foreach, func_get_arg 等】

1.废弃扩展
Ereg 正则表达式
mssql
mysql
sybase_ct

2.废弃的特性
不能使用同名的构造函数
实例方法不能用静态方法的方式调用

3.废弃的函数
方法调用

call_user_method() 
call_user_method_array()

应该采用call_user_func() 和 call_user_func_array()

加密相关函数

mcrypt_generic_end() 
mcrypt_ecb() 
mcrypt_cbc() 
mcrypt_cfb() 
mcrypt_ofb()

注意: PHP7.1 以后mcrypt_*序列函数都将被移除。推荐使用:openssl 序列函数

杂项

set_magic_quotes_runtime 
set_socket_blocking 
Split 
imagepsbbox() 
imagepsencodefont() 
imagepsextendfont() 
imagepsfreefont() 
imagepsloadfont() 
imagepsslantfont() 
imagepstext()

4.废弃的用法
$HTTP_RAW_POST_DATA 变量被移除, 使用php://input来代

ini文件里面不再支持#开头的注释, 使用”;”

移除了ASP格式的支持和脚本语法的支持: <% 和 < script language=php >

1.字符串处理机制修改
含有十六进制字符的字符串不再视为数字, 也不再区别对待.

var_dump("0x123" == "291"); // false
var_dump(is_numeric("0x123")); // false
var_dump("0xe" + "0x1"); // 0
var_dump(substr("f00", "0x1")) // foo

2.整型处理机制修改
Int64支持, 统一不同平台下的整型长度, 字符串和文件上传都支持大于2GB. 64位PHP7字符串长度可以超过2^31次方字节.

// 无效的八进制数字(包含大于7的数字)会报编译错误
$i = 0681; // 老版本php会把无效数字忽略。

// 位移负的位置会产生异常
var_dump(1 » -1);

// 左位移超出位数则返回0
var_dump(1 « 64);// 0

// 右位移超出会返回0或者-1
var_dump(100 » 32);// 0
var_dump(-100 » 32);// -1
3.参数处理机制修改
不支持重复参数命名

function func(b, c) {} ;hui报错

func_get_arg()和func_get_args()这两个方法返回参数当前的值, 而不是传入时的值, 当前的值有可能会被修改

所以需要注意,在函数第一行最好就给记录下来,否则后续有修改的话,再读取就不是传进来的初始值了。

function foo($x) {
    $x++;
    echo func_get_arg(0);
}
foo(1); //返回2

4.foreach修改
foreach()循环对数组内部指针不再起作用

$arr = [1,2,3];
foreach ($arr as &$val) {
    echo current($arr);// php7 全返回0
}

按照值进行循环的时候, foreach是对该数组的拷贝操作

$arr = [1,2,3];
foreach ($arr as $val) {
    unset($arr[1]);
}
var_dump($arr);

最新的php7依旧会打印出[1,2,3]。(ps:7.0.0不行)
老的会打印出[1,3]

按照引用进行循环的时候, 对数组的修改会影响循环

$arr = [1];
foreach ($arr as $val) {
    var_dump($val);
    $arr[1]=2;
}

最新的php7依旧会追加新增元素的循环。(ps:7.0.0不行)

  1. list修改
    不再按照相反的顺序赋值

//$arr将会是[1,2,3]而不是之前的[3,2,1]

list($arr[], $arr[], $arr[]) = [1,2,3];

不再支持字符串拆分功能

// $x = null 并且 $y = null

$str = 'xy';
list($x, $y) = $str;

空的list()赋值不再允许

list() = [123];

list()现在也适用于数组对象

list($a, $b) = (object)new ArrayObject([0, 1]);

6.变量处理机制修改
对变量、属性和方法的间接调用现在将严格遵循从左到右的顺序来解析,而不是之前的混杂着几个特殊案例的情况。 下面这张表说明了这个解析顺序的变化。

这里写图片描述

引用赋值时自动创建的数组元素或者对象属性顺序和以前不同了

$arr = [];
$arr['a'] = &$arr['b'];
$arr['b'] = 1;
// php7: ['a' => 1, 'b' => 1]
// php5: ['b' => 1, 'a' => 1]

7.杂项

1.debug_zval_dump() 现在打印 “int” 替代 “long”, 打印 “float” 替代 “double”

2.dirname() 增加了可选的第二个参数, depth, 获取当前目录向上 depth 级父目录的名称。

3.getrusage() 现在支持 Windows.mktime() and gmmktime() 函数不再接受 is_dst 参数。

4.preg_replace() 函数不再支持 “\e” (PREG_REPLACE_EVAL). 应当使用 preg_replace_callback() 替代。

5.setlocale() 函数不再接受 category 传入字符串。 应当使用 LC_* 常量。

6.exec(), system() and passthru() 函数对 NULL 增加了保护.

7.shmop_open() 现在返回一个资源而非一个int, 这个资源可以传给shmop_size(), shmop_write(), shmop_read(), shmop_close() 和 shmop_delete().

8.为了避免内存泄露,xml_set_object() 现在在执行结束时需要手动清除 $parse。

9.curl_setopt 设置项CURLOPT_SAFE_UPLOAD变更

TRUE 禁用 @ 前缀在 CURLOPT_POSTFIELDS 中发送文件。 意味着 @ 可以在字段中安全得使用了。 可使用 CURLFile作为上传的代替。
PHP 5.5.0 中添加,默认值 FALSE。 PHP 5.6.0 改默认值为 TRUE。. PHP 7 删除了此选项, 必须使用 CURLFile interface 来上传文件。

随机文章