PHP,如何通过零捕获除法?

Cri*_*ian 28 php exception divide-by-zero

我有一个必须动态创建的大型数学表达式.例如,一旦我解析了"某事",结果将是一个字符串,如:"$foo+$bar/$baz";.

所以,为了计算该表达式的结果,我正在使用eval函数......这样的事情:

eval("\$result = $expresion;");
echo "The result is: $result";
Run Code Online (Sandbox Code Playgroud)

这里的问题是,有时我会得到错误,表示存在除零的错误,而且我不知道如何捕获该异常.我尝试过这样的事情:

eval("try{\$result = $expresion;}catch(Exception \$e){\$result = 0;}");
echo "The result is: $result";
Run Code Online (Sandbox Code Playgroud)

要么:

try{
    eval("\$result = $expresion;");
}
catch(Exception $e){
    $result = 0;
}
echo "The result is: $result";
Run Code Online (Sandbox Code Playgroud)

但它不起作用.那么,如果存在除零值,我怎么能避免我的应用程序崩溃?

编辑:

首先,我想澄清一下:表达式是动态构建的,所以如果分母为零,我不能只评估它.那么......关于Mark Ba​​ker的评论,让我举个例子.我的解析器可以构建这样的东西:

"$foo + $bar * ( $baz / ( $foz - $bak ) )"
Run Code Online (Sandbox Code Playgroud)

解析器逐步构建字符串而不用担心变量的值...所以在这种情况下如果$foz == $bak实际上除以零:$baz / ( 0 ).

另一方面,正如皮特建议的那样,我尝试过:

<?php
$a = 5;
$b = 0;

if(@eval(" try{ \$res = $a/$b; } catch(Exception \$e){}") === FALSE)
        $res = 0;
echo "$res\n";
?> 
Run Code Online (Sandbox Code Playgroud)

但它没有打印任何东西.

Mar*_*ker 17

if ($baz == 0.0) {
    echo 'Divisor is 0';
} else {
    ...
}
Run Code Online (Sandbox Code Playgroud)

如果你在evalled表达式中使用用户输入,那么使用eval是非常危险的,而不是使用eval,为什么不在PHPClasses上使用适当的解析器,例如evalmath,并且在除以零时引发一个干净的异常

  • 它是如何动态构建的?它必须由你的"解析器"以某种方式"构造"......在那时你应该能够识别除数是否为0! (4认同)
  • 不...正如我所说,表达式是动态构建的。如果它是一个静态表达式,那就很容易了,但事实并非如此。 (3认同)

Dav*_*d L 9

在 PHP7 上你可以使用DivisionByZeroError

try {
    echo 1/0;
} catch(DivisionByZeroError $e){
    echo "got $e";
} catch(ErrorException $e) {
    echo "got $e";
}
Run Code Online (Sandbox Code Playgroud)


tac*_*one 7

您只需要设置一个错误处理程序以在发生错误的情况下引发异常:

set_error_handler(function () {
    throw new Exception('Ach!');
});

try {
    $result = 4 / 0;
} catch( Exception $e ){
    echo "Divide by zero, I don't fear you!".PHP_EOL;
    $result = 0;
}

restore_error_handler();
Run Code Online (Sandbox Code Playgroud)


Bil*_*win 6

这是另一个解决方案:

<?php

function e($errno, $errstr, $errfile, $errline) {
    print "caught!\n";
}

set_error_handler('e');

eval('echo 1/0;');
Run Code Online (Sandbox Code Playgroud)

看到 set_error_handler()

  • @Pete:是的,但OP问的是零错误除,而不是解析错误.我测试了上面的脚本,它捕获了错误. (4认同)