Вычислить математическое выражение из строки, используя eval
Я хочу вычислить математическое выражение из строки. Я прочитал, что решение этой проблемы заключается в использовании eval(). Но когда я пытаюсь запустить следующий код:
<?php
$ma ="2+10";
$p = eval($ma);
print $p;
?>
Это дает мне следующую ошибку:
Ошибка разбора: синтаксическая ошибка, неожиданный конец $ в C:\xampp\htdocs\eclipseWorkspaceWebDev\MandatoryHandinSite\tester.php(4): код eval () в строке 1
Кто-нибудь знает решение этой проблемы.
10 ответов
Пока я не предлагаю использовать eval
для этого (это не решение), проблема в том, что eval
ожидает полные строки кода, а не только фрагменты.
$ma ="2+10";
$p = eval('return '.$ma.';');
print $p;
Должен делать то, что вы хотите.
Лучшим решением было бы написать токенизатор / парсер для вашего математического выражения. Вот очень простой пример на основе регулярных выражений, чтобы дать вам пример:
$ma = "2+10";
if(preg_match('/(\d+)(?:\s*)([\+\-\*\/])(?:\s*)(\d+)/', $ma, $matches) !== FALSE){
$operator = $matches[2];
switch($operator){
case '+':
$p = $matches[1] + $matches[3];
break;
case '-':
$p = $matches[1] - $matches[3];
break;
case '*':
$p = $matches[1] * $matches[3];
break;
case '/':
$p = $matches[1] / $matches[3];
break;
}
echo $p;
}
Взгляните на это..
Я использую это в бухгалтерской системе, где вы можете написать математические выражения в полях ввода суммы..
Примеры
$Cal = new Field_calculate();
$result = $Cal->calculate('5+7'); // 12
$result = $Cal->calculate('(5+9)*5'); // 70
$result = $Cal->calculate('(10.2+0.5*(2-0.4))*2+(2.1*4)'); // 30.4
Код
class Field_calculate {
const PATTERN = '/(?:\-?\d+(?:\.?\d+)?[\+\-\*\/])+\-?\d+(?:\.?\d+)?/';
const PARENTHESIS_DEPTH = 10;
public function calculate($input){
if(strpos($input, '+') != null || strpos($input, '-') != null || strpos($input, '/') != null || strpos($input, '*') != null){
// Remove white spaces and invalid math chars
$input = str_replace(',', '.', $input);
$input = preg_replace('[^0-9\.\+\-\*\/\(\)]', '', $input);
// Calculate each of the parenthesis from the top
$i = 0;
while(strpos($input, '(') || strpos($input, ')')){
$input = preg_replace_callback('/\(([^\(\)]+)\)/', 'self::callback', $input);
$i++;
if($i > self::PARENTHESIS_DEPTH){
break;
}
}
// Calculate the result
if(preg_match(self::PATTERN, $input, $match)){
return $this->compute($match[0]);
}
// To handle the special case of expressions surrounded by global parenthesis like "(1+1)"
if(is_numeric($input)){
return $input;
}
return 0;
}
return $input;
}
private function compute($input){
$compute = create_function('', 'return '.$input.';');
return 0 + $compute();
}
private function callback($input){
if(is_numeric($input[1])){
return $input[1];
}
elseif(preg_match(self::PATTERN, $input[1], $match)){
return $this->compute($match[0]);
}
return 0;
}
}
Я недавно создал пакет PHP, который обеспечивает math_eval
вспомогательная функция. Он делает именно то, что вам нужно, без необходимости использовать потенциально небезопасно eval
функция.
Вы просто передаете строковую версию математического выражения, и она возвращает результат.
$two = math_eval('1 + 1');
$three = math_eval('5 - 2');
$ten = math_eval('2 * 5');
$four = math_eval('8 / 2');
Вы также можете передать переменные, которые будут заменены при необходимости.
$ten = math_eval('a + b', ['a' => 7, 'b' => 3]);
$fifteen = math_eval('x * y', ['x' => 3, 'y' => 5]);
Решено!
<?php
function evalmath($equation)
{
$result = 0;
// sanitize imput
$equation = preg_replace("/[^a-z0-9+\-.*\/()%]/","",$equation);
// convert alphabet to $variabel
$equation = preg_replace("/([a-z])+/i", "\$$0", $equation);
// convert percentages to decimal
$equation = preg_replace("/([+-])([0-9]{1})(%)/","*(1\$1.0\$2)",$equation);
$equation = preg_replace("/([+-])([0-9]+)(%)/","*(1\$1.\$2)",$equation);
$equation = preg_replace("/([0-9]{1})(%)/",".0\$1",$equation);
$equation = preg_replace("/([0-9]+)(%)/",".\$1",$equation);
if ( $equation != "" ){
$result = @eval("return " . $equation . ";" );
}
if ($result == null) {
throw new Exception("Unable to calculate equation");
}
echo $result;
// return $equation;
}
$a = 2;
$b = 3;
$c = 5;
$f1 = "a*b+c";
$f1 = str_replace("a", $a, $f1);
$f1 = str_replace("b", $b, $f1);
$f1 = str_replace("c", $c, $f1);
evalmath($f1);
/*if ( $equation != "" ){
$result = @eval("return " . $equation . ";" );
}
if ($result == null) {
throw new Exception("Unable to calculate equation");
}
echo $result;*/
?>
Этот метод имеет два основных недостатка:
Безопасность, php скрипт проверяется функцией eval. Это плохо, особенно когда пользователь хочет внедрить вредоносный код.
сложность
Я создал это, проверьте это: Formula Interpreter
Как это работает?
Сначала создайте экземпляр FormulaInterpreter
с формулой и ее параметрами
$formulaInterpreter = new FormulaInterpreter("x + y", ["x" => 10, "y" => 20]);
Использовать execute()
Метод интерпретации формулы. Он вернет результат:
echo $formulaInterpreter->execute();
в одну строку
echo (new FormulaInterpreter("x + y", ["x" => 10, "y" => 20]))->execute();
Примеры
# Formula: speed = distance / time
$speed = (new FormulaInterpreter("distance/time", ["distance" => 338, "time" => 5]))->execute() ;
echo $speed;
#Venezuela night overtime (ordinary_work_day in hours): (normal_salary * days_in_a_work_month)/ordinary_work_day
$parameters = ["normal_salary" => 21000, "days_in_a_work_month" => 30, "ordinary_work_day" => 8];
$venezuelaLOTTTArt118NightOvertime = (new FormulaInterpreter("(normal_salary/days_in_a_work_month)/ordinary_work_day", $parameters))->execute();
echo $venezuelaLOTTTArt118NightOvertime;
#cicle area
$cicleArea = (new FormulaInterpreter("3.1416*(radio*radio)", ["radio" => 10]))->execute();
echo $cicleArea;
О формулах
- Он должен содержать как минимум два операнда и оператор.
- Имя операнда может быть в верхнем или нижнем регистре.
- К настоящему времени математические функции как sin, cos, pow… не включены. Я работаю, чтобы включить их.
- Если ваша формула неверна, вы получите сообщение об ошибке типа: Ошибка, ваша формула (single_variable) недействительна.
- Значения параметров должны быть числовыми.
Вы можете улучшить его, если хотите!
Чтобы найти золотую середину между опасностями eval и безграничными вычислительными возможностями, я предлагаю проверять ввод только на числа, операторы и скобки:
if (preg_match('/^[0-9\+\-\*\/\(\)\.]+$/', $mathString)) {
$value = eval('return
' . $mathString . ';');
} else {
throw new \Exception('Invalid calc() value: ' . $mathString);
}
Он по-прежнему прост в использовании, но относительно экономичен. И он может обрабатывать любые базовые математические вычисления, такие как
Использование функции eval
protected function getStringArthmeticOperation($value, $deduct)
{
if($value > 0){
$operator = '-';
}else{
$operator = '+';
}
$mathStr = '$value $operator $deduct';
eval("\$mathStr = \"$mathStr\";");
$userAvailableUl = eval('return '.$mathStr.';');
return $userAvailableUl;
}
$this->getStringArthmeticOperation(3, 1); //2
eval
Оценивает данный код какPHP
, Это означает, что он будет выполнять данный параметр в виде фрагмента кода PHP.
Чтобы исправить свой код, используйте это:
$ma ="print (2+10);";
eval($ma);
Выражение eval'd должно заканчиваться на ";"
Попробуй это:
$ma ="2+10;";
$p = eval($ma);
print $p;
Кстати, это выходит за рамки, но функция 'eval' не будет возвращать значение выражения. eval('2+10') не вернет 12. Если вы хотите, чтобы он возвратил 12, вы должны eval('return 2+10;');