PHP 7 测试用例(转)

简介: 性能改善:PHP 7高达两倍快的PHP 5.6 显著减少内存使用 抽象语法树 一致的64位支持 改进的异常层次结构 许多转化为异常致命错误 安全随机数发生器 删除旧的和不支持的SAPIs和扩展 空合并运算符(?) 返回和标量类型声明 匿名类 零成本断言   1 、测试用例一...

性能改善:PHP 7高达两倍快的PHP 5.6

显著减少内存使用

抽象语法树

一致的64位支持

改进的异常层次结构

许多转化为异常致命错误

安全随机数发生器

删除旧的和不支持的SAPIs和扩展

空合并运算符(?)

返回和标量类型声明

匿名类

零成本断言

 

1 、测试用例一:

生成五十万个数组,并查询五十万次 key 是否存在

<?php

$a = array();

for($i=0;$i<500000;$i++){

$a[$i] = $i;

}

foreach($a as $i)

{

array_key_exists($i, $a);

}

?>

测试结果如下:

 time php test.php

php test.php

0.60s user

0.05s system

98% cpu

0.667 total

 time /usr/local/php7/bin/php test.php

/usr/local/php7/bin/php test.php

0.05s user

0.02s system

92% cpu

0.073 total

PHP7 速度是 PHP5.5 的 9 倍

2 、测试用例二:

生成五十万个数组,并查询五十万次 value 是否存在

<?php

$a = array();

for($i=0;$i<10000;$i++){

$a[$i] = $i;

}

foreach($a as $i)

{

array_key_exists($i, $a);

}

?>

 time php test.php

php test.php

0.79s user

0.01s system

99% cpu

0.809 total

 time /usr/local/php7/bin/php test.php

/usr/local/php7/bin/php test.php

0.08s user

0.01s system

97% cpu

0.091 total

PHP7 速度是 PHP5.5 的 8.7 倍

3 、测试用例三:

示例与结果摘自鸟哥博客。以 Wordpress 为基础,测试 PHP7 和 HHVM3.2 。用Apache 的 ab 测试工具。 100 个并发 , 10000 个请求。测试前都会用 100 个请求预热。

PHP7 结果如下:

Concurrency Level:      100

Time taken for tests:   38.726 seconds

Complete requests:      10000

Failed requests:        0

Write errors:           0

Total transferred:      89290000 bytes

HTML transferred:       86900000 bytes

Requests per second:    258.22 [#/sec] (mean)

Time per request:       387.260 [ms] (mean)

Time per request:       3.873 [ms] (mean, across all concurrent requests)

Transfer rate:          2251.64 [Kbytes/sec] received

HHVM-3.2

HHVM 结果如下:

Document Path:          /wordpress/

Document Length:        8690 bytes

Concurrency Level:      100

Time taken for tests:   43.296 seconds

Complete requests:      10000

Failed requests:        0

Write errors:           0

Total transferred:      89260000 bytes

HTML transferred:       86900000 bytes

Requests per second:    230.97 [#/sec] (mean)

Time per request:       432.957 [ms] (mean)

Time per request:       4.330 [ms] (mean, across all concurrent requests)

Transfer rate:          2013.31 [Kbytes/sec] received

PHP7 – 258.22 QPS HHVM – 230.97 QPS

四、新特性

1 、标量类型声明

有两种模式 : 强制 ( 默认 ) 和 严格模式。 现在可以使用下列类型参数(无论用强制模式还是严格模式): 字符串 (string), 整数 (int), 浮点数 (float), 以及布尔值 (bool) 。它们扩充了 PHP5 中引入的其他类型:类名,接口,数组和 回调类型。在旧版中,函数的参数申明只能是 (Array $arr) 、 (CLassName $obj) 等,基本类型比如 Int , String等是不能够被申明的

<?php

function check(int $bool){

var_dump($bool);

}

check(1);

check(true);

?>

若无强制类型转换,会输入 int(1)bool(true) 。转换后会输出 bool(true) bool(true)

2 、返回值类型声明

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

)

3 、 null 合并运算符

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

旧版: isset($_GET[‘id']) ? $_GET[id] : err;

新版: $_GET['id'] ?? 'err';

4 、太空船操作符(组合比较符)

太空船操作符用于比较两个表达式。当 $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

?>

5 、通过 define() 定义常量数组

<?php

define('ANIMALS', ['dog', 'cat', 'bird']);

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

?>

6 、匿名类

现在支持通过 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());

7 、 Unicode codepoint 转译语法

这接受一个以 16 进制形式的 Unicode codepoint ,并打印出一个双引号或 heredoc包围的 UTF-8 编码格式的字符串。 可以接受任何有效的 codepoint ,并且开头的 0 是可以省略的。

<?php

echo “\u{9876}”

?>

旧版输出: \u{9876}

新版输入:顶

8 、 Closure::call()

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

<?php

class Test{public $name = "lixuan";}

//PHP7 和 PHP5.6 都可以

$getNameFunc = function(){return $this->name;};

$name = $getNameFunc->bindTo(new Test, 'Test');

echo $name();

//PHP7 可以 ,PHP5.6 报错

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

echo $getX->call(new Test);

9 、为 unserialize() 提供过滤

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

<?php

// 将所有对象分为 __PHP_Incomplete_Class 对象

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

// 将所有对象分为 __PHP_Incomplete_Class 对象 除了 ClassName1 和ClassName2

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

// 默认行为,和 unserialize($foo) 相同

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

10 、 IntlChar

新增加的 IntlChar 类旨在暴露出更多的 ICU 功能。这个类自身定义了许多静态方法用于操作多字符集的 unicode 字符。 Intl 是 Pecl 扩展,使用前需要编译进 PHP 中,也可 apt-get/yum/port install php5-intl

<?php

printf('%x', IntlChar::CODEPOINT_MAX);

echo IntlChar::charName('@');

var_dump(IntlChar::ispunct('!'));

?>

以上例程会输出:

10ffff

COMMERCIAL AT

bool(true)

11 、预期

预期是向后兼用并增强之前的 assert() 的方法。 它使得在生产环境中启用断言为零成本,并且提供当断言失败时抛出特定异常的能力。 老版本的 API 出于兼容目的将继续被维护, assert() 现在是一个语言结构,它允许第一个参数是一个表达式,而不仅仅是一个待计算的 string 或一个待测试的 boolean 。

<?php

ini_set('assert.exception', 1);

class CustomError extends AssertionError {}

assert(false, new CustomError('Some error message'));

?>

以上例程会输出:

Fatal error: Uncaught CustomError: Some error message

12 、 Group use declarations

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

<?php

//PHP7 之前

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;

 

// PHP7 之后

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};

?>

13 、 intdiv()

接收两个参数作为被除数和除数,返回他们相除结果的整数部分。

<?php

var_dump(intdiv(7, 2));

?>

输出 int(3)

14 、 CSPRNG

新增两个函数 : random_bytes() and random_int(). 可以加密的生产被保护的整数和字符串。我这蹩脚的翻译,总之随机数变得安全了。

andom_bytes — 加密生存被保护的伪随机字符串

random_int — 加密生存被保护的伪随机整数

15 、 preg_replace_callback_array()

新增了一个函数 preg_replace_callback_array() ,使用该函数可以使得在使用preg_replace_callback() 函数时代码变得更加优雅。在 PHP7 之前,回调函数会调用每一个正则表达式,回调函数在部分分支上是被污染了。

16 、 Session options

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

比如,把 cache_limiter 设置为私有的,同时在阅读完 session 后立即关闭。

<?php

session_start([

'cache_limiter' => 'private',

'read_and_close' => true,

]);

?>

17 、生成器的返回值

在 PHP5.5 引入生成器的概念。生成器函数每执行一次就得到一个 yield 标识的值。在 PHP7 中,当生成器迭代完成后,可以获取该生成器函数的返回值。通过Generator::getReturn() 得到。

<?php

function generator() {

yield 1;

yield 2;

yield 3;

return "a";

}

$generatorClass = ("generator")();

foreach ($generatorClass as $val) {

echo $val.” “;

}

echo $generatorClass->getReturn();

?>

输出为: 1 2 3 a

18 、生成器中引入其他生成器

在生成器中可以引入另一个或几个生成器,只需要写 yield from functionName1

<?php

function generator1(){

yield 1;

yield 2;

yield from generator2();

yield from generator3();

}

function generator2(){

yield 3;

yield 4;

}

function generator3(){

yield 5;

yield 6;

}

foreach (generator1() as $val){

echo $val, " ";

}

?>

输出: 1 2 3 4 5 6

五、不兼容性

1 、 foreach 不再改变内部数组指针

在 PHP7 之前,当数组通过 foreach 迭代时,数组指针会移动。现在开始,不再如此,见下面代码。

<?php

$array = [0, 1, 2];

foreach ($array as &$val) {

var_dump(current($array));

}

?>

PHP5 输出:

int(1)

int(2)

bool(false)

PHP7 输出:

int(0)

int(0)

int(0)

2 、 foreach 通过引用遍历时,有更好的迭代特性

当使用引用遍历数组时,现在 foreach 在迭代中能更好的跟踪变化。例如,在迭代中添加一个迭代值到数组中,参考下面的代码:

<?php

$array = [0];

foreach ($array as &$val) {

var_dump($val);

$array[1] = 1;

}

?>

PHP5 输出:

int(0)

PHP7 输出:

int(0)

int(1)

3 、十六进制字符串不再被认为是数字

含十六进制字符串不再被认为是数字

<?php

var_dump("0x123" == "291");

var_dump(is_numeric("0x123"));

var_dump("0xe" + "0x1");

var_dump(substr("foo", "0x1"));

?>

PHP5 输出:

bool(true)

bool(true)

int(15)

string(2) "oo"

PHP7 输出:

bool(false)

bool(false)

int(0)

Notice: A non well formed numeric value encountered in /tmp/test.php on line 5

string(3) "foo"

4 、 PHP7 中被移除的函数

被移除的函数列表如下:

call_user_func() 和 call_user_func_array() 从 PHP 4.1.0 开始被废弃。

已废弃的 mcrypt_generic_end() 函数已被移除,请使用 mcrypt_generic_deinit() 代替。

已废弃的 mcrypt_ecb(), mcrypt_cbc(), mcrypt_cfb() 和 mcrypt_ofb() 函数已被移除。

set_magic_quotes_runtime(), 和它的别名 magic_quotes_runtime() 已被移除 . 它们在 PHP 5.3.0 中已经被废弃 , 并且 在 in PHP 5.4.0 也由于魔术引号的废弃而失去功能。

已废弃的 set_socket_blocking() 函数已被移除,请使用 stream_set_blocking() 代替。

dl() 在 PHP-FPM 不再可用,在 CLI 和 embed SAPIs 中仍可用。

GD 库中下列函数被移除: imagepsbbox() 、 imagepsencodefont() 、imagepsextendfont() 、 imagepsfreefont() 、 imagepsloadfont() 、 imagepsslantfont()、 imagepstext()

在配置文件 php.ini 中, always_populate_raw_post_data 、 asp_tags 、xsl.security_prefs 被移除了。

5 、 new 操作符创建的对象不能以引用方式赋值给变量

new 操作符创建的对象不能以引用方式赋值给变量

<?php

class C {}

$c =& new C;

?>

PHP5 输出:

Deprecated: Assigning the return value of new by reference is deprecated in /tmp/test.php on line 3

PHP7 输出:

Parse error: syntax error, unexpected 'new' (T_NEW) in /tmp/test.php on line 3

6 、移除了 ASP 和 script PHP 标签

使用类似 ASP 的标签,以及 script 标签来区分 PHP 代码的方式被移除。 受到影响的标签有: <% %> 、 <%= %> 、 <script language="php"> </script>

7 、从不匹配的上下文发起调用

在不匹配的上下文中以静态方式调用非静态方法, 在 PHP 5.6 中已经废弃, 但是在PHP 7.0 中, 会导致被调用方法中未定义 $this 变量,以及此行为已经废弃的警告。

<?php

class A {

public function test() { var_dump($this); }

}

// 注意:并没有从类 A 继承

class B {

public function callNonStaticMethodOfA() { A::test(); }

}

(new B)->callNonStaticMethodOfA();

?>

PHP5 输出:

Deprecated: Non-static method A::test() should not be called statically, assuming $this from incompatible context in /tmp/test.php on line 8

object(B)#1 (0) {

}

PHP7 输出:

Deprecated: Non-static method A::test() should not be called statically in /tmp/test.php on line 8

Notice: Undefined variable: this in /tmp/test.php on line 3

NULL

8 、在数值溢出的时候,内部函数将会失败

将浮点数转换为整数的时候,如果浮点数值太大,导致无法以整数表达的情况下,在之前的版本中,内部函数会直接将整数截断,并不会引发错误。 在 PHP 7.0 中,如果发生这种情况,会引发 E_WARNING 错误,并且返回 NULL 。

9 、 JSON 扩展已经被 JSOND 取代

JSON 扩展已经被 JSOND 扩展取代。 对于数值的处理,有以下两点需要注意的:第一,数值不能以点号( . )结束 (例如,数值 34. 必须写作 34.0 或 34 )。 第二,如果使用科学计数法表示数值, e 前面必须不是点号( . ) (例如, 3.e3 必须写作3.0e3 或 3e3 )。

10 、 INI 文件中 # 注释格式被移除

在配置文件 INI 文件中,不再支持以 # 开始的注释行, 请使用 ; (分号)来表示注释。 此变更适用于 php.ini 以及用 parse_ini_file() 和 parse_ini_string() 函数来处理的文件。

11 、 $HTTP_RAW_POST_DATA 被移除

不再提供 $HTTP_RAW_POST_DATA 变量。 请使用 php://input 作为替代。

12 、 yield 变更为右联接运算符

在使用 yield 关键字的时候,不再需要括号, 并且它变更为右联接操作符,其运算符优先级介于 print 和 => 之间。 这可能导致现有代码的行为发生改变。可以通过使用括号来消除歧义。

<?php

echo yield -1;

// 在之前版本中会被解释为:

echo (yield) - 1;

// 现在,它将被解释为:

echo yield (-1);

yield $foo or die;

// 在之前版本中会被解释为:

yield ($foo or die);

// 现在,它将被解释为:

(yield $foo) or die;

?>

PHP

目录
相关文章
原生php实现列表接口+分页接口+排序接口组合使用+包括测试数据(不加任何封装)
原生php实现列表接口+分页接口+排序接口组合使用+包括测试数据(不加任何封装)
原生php实现列表接口+分页接口+排序接口组合使用+包括测试数据(不加任何封装)
|
2月前
|
测试技术 PHP 开发工具
php性能监测模块XHProf安装与测试
【10月更文挑战第13天】php性能监测模块XHProf安装与测试
33 0
|
3月前
|
设计模式 SQL 安全
PHP中的设计模式:单例模式的深入探索与实践在PHP的编程实践中,设计模式是解决常见软件设计问题的最佳实践。单例模式作为设计模式中的一种,确保一个类只有一个实例,并提供全局访问点,广泛应用于配置管理、日志记录和测试框架等场景。本文将深入探讨单例模式的原理、实现方式及其在PHP中的应用,帮助开发者更好地理解和运用这一设计模式。
在PHP开发中,单例模式通过确保类仅有一个实例并提供一个全局访问点,有效管理和访问共享资源。本文详细介绍了单例模式的概念、PHP实现方式及应用场景,并通过具体代码示例展示如何在PHP中实现单例模式以及如何在实际项目中正确使用它来优化代码结构和性能。
52 2
|
3月前
|
设计模式 人工智能 算法
PHP中的设计模式:策略模式的深入解析与实践软件测试中的人工智能革命:提升效率与准确性的新篇章
在PHP开发中,理解并运用设计模式是提升代码质量和可维护性的重要途径。本文聚焦于策略模式(Strategy Pattern),一种行为型设计模式,它允许在运行时选择算法或业务规则。通过本文,我们将深入探讨策略模式的定义、结构、使用场景以及如何在PHP项目中有效地实现和利用策略模式。不同于性能优化等技术性摘要,本文着重于提供对策略模式全面而实用的理解,助力开发者编写出更加灵活和可扩展的应用程序。 本文深入探讨了人工智能在软件测试领域的应用,揭示了其如何显著提高测试过程的效率和准确性。通过实际案例分析,展示了AI技术在自动化测试、缺陷检测及结果分析中的关键作用,并讨论了实施AI测试策略时面临的挑
26 3
|
5月前
|
测试技术 PHP 开发者
原生php单元测试示例
通过上面的示例,我们可以看到,即使在缺乏专门测试框架的情况下,使用原生PHP代码进行基本的单元测试也是完全可行的。当然,对于更复杂的项目,利用像PHPUnit这样的专业工具将带来更多的便利和高级功能。不过,理解单元测试的基本概念和能够手工编写测试是每个PHP开发者的宝贵技能。
39 4
|
5月前
|
测试技术 PHP 开发者
原生php单元测试示例
通过上面的示例,我们可以看到,即使在缺乏专门测试框架的情况下,使用原生PHP代码进行基本的单元测试也是完全可行的。当然,对于更复杂的项目,利用像PHPUnit这样的专业工具将带来更多的便利和高级功能。不过,理解单元测试的基本概念和能够手工编写测试是每个PHP开发者的宝贵技能。
24 1
|
5月前
|
数据库
基于PHP+MYSQL开发制作的趣味测试网站源码
基于PHP+MYSQL开发制作的趣味测试网站源码。可在后台提前设置好缘分, 自己手动在数据库里修改数据,数据库里有就会优先查询数据库的信息, 没设置的话第一次查询缘分都是非常好的 95-99,第二次查就比较差 , 所以如果要你女朋友查询你的名字觉得很好 那就得是她第一反应是查和你的缘分, 如果查的是别人,那不好意思,第二个可能是你。
72 3
|
5月前
|
测试技术 PHP 开发者
原生php单元测试示例
通过上面的示例,我们可以看到,即使在缺乏专门测试框架的情况下,使用原生PHP代码进行基本的单元测试也是完全可行的。当然,对于更复杂的项目,利用像PHPUnit这样的专业工具将带来更多的便利和高级功能。不过,理解单元测试的基本概念和能够手工编写测试是每个PHP开发者的宝贵技能。
28 0
|
PHP 数据库
对PHP源码的测试
对PHP源码的测试