当前位置: 移动技术网 > IT编程>开发语言>PHP > 你所不知的 PHP 断言(assert)

你所不知的 PHP 断言(assert)

2019年06月03日  | 移动技术网IT编程  | 我要评论

馕,成人日漫,丹61库珀

php 中的断言常用于调试,检查一个表达式或语句是否为 false。本文带你重新认识 php assert() 函数的神(qi)通(yin)广(ji)大(qiao)。

本文基于 php version 7.1.28

什么是断言

编写程序时,常会做出一定的假设,那断言就是用来捕获假设的异常,我们也可以认为断言是异常的一种特殊形式。

断言一般用于程序执行结构的判断,不可让断言处理业务流程。用的最多的场景就是单元测试,一般的单元测试框架都采用了断言。

assert(1 == 2);

// 运行结果:
// warning: assert(): assert(1 == 2) failed in /users/shocker/desktop/demo.php on line 25

php 中的断言

在 php 中,采用 函数对表达式进行断言。

// php 5
assert ( mixed $assertion [, string $description ] ) : bool

// php 7
assert ( mixed $assertion [, throwable $exception ] ) : bool

传统的断言方式 (php 5 & 7)

参数 assertion 既支持表达式,也支持表达式字符串(某些特定的场景会用到,比如判断某个字符串表达式是否合法)

如果 assertion 是字符串,它将会被 assert() 当做 php 代码来执行。assertion 是字符串的优势是当禁用断言时它的开销会更小,并且在断言失败时消息会包含 assertion 表达式

断言这个功能应该只被用来调试。你应该用于完整性检查时测试条件是否始终应该为 true,来指示某些程序错误,或者检查具体功能的存在(类似扩展函数或特定的系统限制和功能)。

断言不应该用于普通运行时操作,类似输入参数的检查。作为一个经验法则,在断言禁用时你的代码也应该能够正确地运行。

使用示例:

function my_assert_handler($file, $line, $code, $desc)
{
    echo "assertion failed:
    file '{$file}'
    line '{$line}'
    code '{$code}'
    desc '{$desc}'
";
}

// 设置回调函数
assert_options(assert_callback, 'my_assert_handler');

// 让一则断言失败
assert('1 == 2', '1 不可能等于 2');

运行结果:

assertion failed:
    file '/users/shocker/desktop/demo.php'
    line '29'
    code '1 == 2'
    desc '1 不可能等于 2'

支持异常的断言 (仅 php 7)

在 php 7 中,assert() 是一个语言结构,允许在不同环境中生效不同的措施,具体可见 配置。

另外,还支持通过 assertionerror 捕获错误。

使用示例:

assert_options(assert_exception, 1); // 在断言失败时产生异常

try {
    // 用 assertionerror 异常替代普通字符串
    assert(true == false, new assertionerror('true is not false!'));
} catch (throwable $e) {
    echo $e->getmessage();
}

运行结果:

true is not false!

对断言行为进行控制

php 支持 函数对断言进行配置,也可用 ini 进行设置

以下配置中,常量标志用于 assert_options() 函数进行配置,ini 设置用于 ini_set() 函数设置,效果一样

标志 ini 设置 默认值 描述
assert_active assert.active "1" 启用 assert() 断言
assert_warning assert.warning "1" 为每个失败的断言产生一个 php 警告(warning)
assert_bail assert.bail "0" 在断言失败时中止执行
assert_quiet_eval assert.quiet_eval "0" 在断言表达式求值时禁用 error_reporting
assert_callback assert.callback null 断言失败时调用该回调函数
assert_exception assert.exception "0" 在断言失败时产生 assertionerror 异常 (自 php 7.0.0 起有效)

zend.assertions 是个特殊的配置(php >= 7.0.0 支持),控制不同运行环境下断言的行为,仅可用 ini_set() 进行设置。并且,设置了1就不能再设置为-1,反之亦然,其他不受限。

  • 1: 编译代码,并执行(开发模式)
  • 0: 编辑代码,但运行时跳过
  • -1: 不编译代码(生产模式)

版本的不兼容

  • php >= 5.4.8,description 可作为第四个参数提供给 assert_callback 模式里的回调函数

  • 在 php 5 中,参数 assertion 必须是可执行的字符串,或者运行结果为布尔值的表达式

  • 在 php 7 中,参数 assertion 可以是任意表达式,并用其运算结果作为断言的依据

  • 在 php 7 中,参数 exception 可以是个 throwable 对象,用于捕获表达式运行错误或断言结果为失败。(当然 需开启)

  • php >= 7.0.0,支持 zend.assertionsassert.exception 相关配置及其特性

  • php >= 7.2 版本开始,参数 assertion 不再支持字符串

    详见 php 7.2.x 中废弃的功能

    deprecated: assert(): calling assert() with a string argument is deprecated

应用场景

调试输出

先看示例:

assert('1 == 2', '1 不可能等于 2');

运行结果:

warning: assert(): 1 不可能等于 2: "1 == 2" failed in /users/shocker/desktop/demo.php on line 10

类似于:

$expression = 1 == 2;
if (!($expression)) {
    echo "1 不可能等于 2\n";
    var_dump($expression);
    echo __file__ . "\n";
}

但是,我们无法得知 $expression 的具体表达式,也无法得知具体的执行行数。

单元测试

function arraysum(array $nums) {
    $sum = 0;
    foreach ($nums as $n) {
        $sum += $n;
    }

    return $sum;
}

assert(arraysum([1, 2, 3]) == 6, 'arraysum() 测试不通过:');
assert(is_numeric(arraysum([1, 2, 3])), 'arraysum() 测试不通过:');

是不是跟我们用 phpunit 写单元测试很像

如对本文有疑问,请在下面进行留言讨论,广大热心网友会与你互动!! 点击进行留言回复

相关文章:

验证码:
移动技术网