用php做计算,如何用PHP制作计算器?

根据你的需要,我建议调车场算法..它很容易实现,并且运行得很好。

下面是我刚才提到的一个例子:要旨.

下面是代码副本/粘贴到一个块中:

表达定义:class Parenthesis extends TerminalExpression {

protected $precidence = 7;

public function operate(Stack $stack) {

}

public function getPrecidence() {

return $this->precidence;

}

public function isNoOp() {

return true;

}

public function isParenthesis() {

return true;

}

public function isOpen() {

return $this->value == '(';

}}class Number extends TerminalExpression {

public function operate(Stack $stack) {

return $this->value;

}}abstract class Operator extends TerminalExpression {

protected $precidence = 0;

protected $leftAssoc = true;

public function getPrecidence() {

return $this->precidence;

}

public function isLeftAssoc() {

return $this->leftAssoc;

}

public function isOperator() {

return true;

}}class Addition extends Operator {

protected $precidence = 4;

public function operate(Stack $stack) {

return $stack->pop()->operate($stack) + $stack->pop()->operate($stack);

}}class Subtraction extends Operator {

protected $precidence = 4;

public function operate(Stack $stack) {

$left = $stack->pop()->operate($stack);

$right = $stack->pop()->operate($stack);

return $right - $left;

}}class Multiplication extends Operator {

protected $precidence = 5;

public function operate(Stack $stack) {

return $stack->pop()->operate($stack) * $stack->pop()->operate($stack);

}}class Division extends Operator {

protected $precidence = 5;

public function operate(Stack $stack) {

$left = $stack->pop()->operate($stack);

$right = $stack->pop()->operate($stack);

return $right / $left;

}}class Power extends Operator {

protected $precidence=6;

public function operate(Stack $stack) {

$left = $stack->pop()->operate($stack);

$right = $stack->pop()->operate($stack);

return pow($right, $left);

}}abstract class TerminalExpression {

protected $value = '';

public function __construct($value) {

$this->value = $value;

}

public static function factory($value) {

if (is_object($value) && $value instanceof TerminalExpression) {

return $value;

} elseif (is_numeric($value)) {

return new Number($value);

} elseif ($value == '+') {

return new Addition($value);

} elseif ($value == '-') {

return new Subtraction($value);

} elseif ($value == '*') {

return new Multiplication($value);

} elseif ($value == '/') {

return new Division($value);

} elseif ($value == '^') {

return new Power($value);

} elseif (in_array($value, array('(', ')'))) {

return new Parenthesis($value);

}

throw new Exception('Undefined Value ' . $value);

}

abstract public function operate(Stack $stack);

public function isOperator() {

return false;

}

public function isParenthesis() {

return false;

}

public function isNoOp() {

return false;

}

public function render() {

return $this->value;

}}

堆栈(非常简单的实现):class Stack {

protected $data = array();

public function push($element) {

$this->data[] = $element;

}

public function poke() {

return end($this->data);

}

public function pop() {

return array_pop($this->data);

}}

最后,Executor类:class Math {

protected $variables = array();

public function evaluate($string) {

$stack = $this->parse($string);

return $this->run($stack);

}

public function parse($string) {

$tokens = $this->tokenize($string);

$output = new Stack();

$operators = new Stack();

foreach ($tokens as $token) {

$token = $this->extractVariables($token);

$expression = TerminalExpression::factory($token);

if ($expression->isOperator()) {

$this->parseOperator($expression, $output, $operators);

} elseif ($expression->isParenthesis()) {

$this->parseParenthesis($expression, $output, $operators);

} else {

$output->push($expression);

}

}

while (($op = $operators->pop())) {

if ($op->isParenthesis()) {

throw new RuntimeException('Mismatched Parenthesis');

}

$output->push($op);

}

return $output;

}

public function registerVariable($name, $value) {

$this->variables[$name] = $value;

}

public function run(Stack $stack) {

while (($operator = $stack->pop()) && $operator->isOperator()) {

$value = $operator->operate($stack);

if (!is_null($value)) {

$stack->push(TerminalExpression::factory($value));

}

}

return $operator ? $operator->render() : $this->render($stack);

}

protected function extractVariables($token) {

if ($token[0] == '$') {

$key = substr($token, 1);

return isset($this->variables[$key]) ? $this->variables[$key] : 0;

}

return $token;

}

protected function render(Stack $stack) {

$output = '';

while (($el = $stack->pop())) {

$output .= $el->render();

}

if ($output) {

return $output;

}

throw new RuntimeException('Could not render output');

}

protected function parseParenthesis(TerminalExpression $expression, Stack $output, Stack $operators) {

if ($expression->isOpen()) {

$operators->push($expression);

} else {

$clean = false;

while (($end = $operators->pop())) {

if ($end->isParenthesis()) {

$clean = true;

break;

} else {

$output->push($end);

}

}

if (!$clean) {

throw new RuntimeException('Mismatched Parenthesis');

}

}

}

protected function parseOperator(TerminalExpression $expression, Stack $output, Stack $operators) {

$end = $operators->poke();

if (!$end) {

$operators->push($expression);

} elseif ($end->isOperator()) {

do {

if ($expression->isLeftAssoc() && $expression->getPrecidence() <= $end->getPrecidence()) {

$output->push($operators->pop());

} elseif (!$expression->isLeftAssoc() && $expression->getPrecidence() getPrecidence()) {

$output->push($operators->pop());

} else {

break;

}

} while (($end = $operators->poke()) && $end->isOperator());

$operators->push($expression);

} else {

$operators->push($expression);

}

}

protected function tokenize($string) {

$parts = preg_split('((\d+|\+|-|\(|\)|\*|/)|\s+)', $string, null, PREG_SPLIT_NO_EMPTY | PREG_SPLIT_DELIM_CAPTURE);

$parts = array_map('trim', $parts);

return $parts;

}}

它的工作方式是首先标记输入(基于Word边界和令牌)。然后,运行分流场算法将输入转换为RPN(反向波兰符号)堆栈。然后,这只是一个执行堆栈的问题。下面是一个简单的例子:$math = new Math();$answer = $math->evaluate('(2 + 3) * 4');var_dump($answer);// int(20)$answer = $math->evaluate('1 + 2 * ((3 + 4) * 5 + 6)');var_dump($answer);// int(83)$answer = $math->evaluate('(1 + 2) * (3 + 4) * (5 + 6)');var_dump($answer);// int(231)$math->registerVariable('a', 4);$answer = $math->evaluate('($a + 3) * 4');var_dump($answer);// int(28)$math->registerVariable('a', 5);$answer = $math->evaluate('($a + $a) * 4');var_dump($answer);// int(40)

现在,这个示例比您可能需要的要复杂得多。原因是它还处理分组和运算符优先级。但这是一个不使用EVAL并支持变量的运行算法的很好的例子.

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值