表达式求值(二叉树方法/C++语言描述)(四)
代码清单
// binarytree.h
#ifndef BINARYTREE_H
#define BINARYTREE_H template <typename T> class BinaryTree; template <typename T>
class BinaryTreeNode
{
public:
friend class BinaryTree<T>;
friend class Calculator;
T _data; private:
BinaryTreeNode<T> * _leftChild;
BinaryTreeNode<T> * _rightChild;
}; template <typename T>
class BinaryTree
{
public:
BinaryTree()
{
_root = nullptr;
} ~BinaryTree()
{
destory();
} void preOrder()
{
preOrder(_root);
} void inOrder()
{
inOrder(_root);
} void postOrder()
{
postOrder(_root);
} protected:
BinaryTreeNode<T> * _root; void destory()
{
if (_root)
{
destory(_root);
delete _root;
}
} void destory(BinaryTreeNode<T> * node)
{
if (node)
{
destory(node->_leftChild);
destory(node->_rightChild);
// visit binary tree data
if (node->_leftChild)
{
delete node->_leftChild;
}
if (node->_rightChild)
{
delete node->_rightChild;
}
}
} virtual void preOrder(BinaryTreeNode<T> * node)
{
if (node)
{
// visit binary tree data
preOrder(node->_leftChild);
preOrder(node->_rightChild);
}
} virtual void inOrder(BinaryTreeNode<T> * node)
{
if (node)
{
inOrder(node->_leftChild);
// visit binary tree data
inOrder(node->_rightChild);
}
} virtual void postOrder(BinaryTreeNode<T> * node)
{
if (node)
{
postOrder(node->_leftChild);
postOrder(node->_rightChild);
// visit binary tree data
}
}
}; #endif // BINARYTREE_H // calculator.h
#ifndef CALCULATOR_H
#define CALCULATOR_H #include <cassert>
#include <string>
#include <stack>
#include <iostream>
#include "binarytree.h" using namespace std; typedef enum
{
BEGIN,
NUMBER,
OPERATOR,
LEFT_BRAC,
RIGHT_BRAC
} TokenType; class Token
{
public:
TokenType _type;
union
{
char op;
double num;
} _data;
}; class Calculator : public BinaryTree<Token>
{
public:
void parseExpression(string expression) throw(string);
double calculate(); private:
stack<char> _stkOperators;
stack<BinaryTreeNode<Token> *> _stkNodes;
stack<double> _stkNumbers; static int priority(char op);
static double calculate(double d1, char op, double d2) throw(string); void postOrder(BinaryTreeNode<Token> * node);
void calculateStack() throw(string);
void dealWithNumber(char *&pToken) throw(string);
void dealWithOperator(char *&pToken) throw(string);
void dealWithLeftBrac(char *&pToken) throw(string);
void dealWithRightBrac(char *&pToken) throw(string);
}; #endif // CALCULATOR_H // calculator.cpp
#include "calculator.h" int Calculator::priority(char op)
{
assert(op == '+' || op == '-' || op == '*' || op == '/' || op == '('); if (op == '+' || op == '-')
{
return ;
}
else if (op == '*' || op == '/')
{
return ;
}
else
{
return ;
}
} double Calculator::calculate(double d1, char op, double d2) throw(string)
{
assert(op == '+' || op == '-' || op == '*' || op =='/'); cout << d1 << op << d2 << endl; if (op == '+')
{
return d1 + d2;
}
else if (op == '-')
{
return d1 - d2;
}
else if (op == '*')
{
return d1 * d2;
}
else
{
if (!d2)
{
throw string("divided by 0");
}
return d1 / d2;
}
} void Calculator::calculateStack() throw(string)
{
BinaryTreeNode<Token> * node = new BinaryTreeNode<Token>();
assert(node);
node->_data._type = OPERATOR;
node->_data._data.op = _stkOperators.top();
_stkOperators.pop();
assert(!_stkNodes.empty());
node->_rightChild = _stkNodes.top();
_stkNodes.pop();
assert(!_stkNodes.empty());
node->_leftChild = _stkNodes.top();
_stkNodes.pop();
_stkNodes.push(node);
} void Calculator::parseExpression(string expression) throw(string)
{
destory();
while (!_stkNodes.empty())
{
_stkNodes.pop();
}
while (!_stkOperators.empty())
{
_stkOperators.pop();
}
TokenType lastToken = BEGIN; char * pToken = &expression[];
while (*pToken)
{
switch (lastToken)
{
case BEGIN:
if (*pToken == '(')
{
// an expression begin with a left bracket
dealWithLeftBrac(pToken);;
lastToken = LEFT_BRAC;
}
else
{
// or a number
dealWithNumber(pToken);
lastToken = NUMBER;
}
break;
case NUMBER:
// after a number
if (*pToken == ')')
{
// it may be a right bracket
dealWithRightBrac(pToken);
lastToken = RIGHT_BRAC;
}
else
{
// it may be an operator
dealWithOperator(pToken);
lastToken = OPERATOR;
}
break;
case OPERATOR:
case LEFT_BRAC:
// after an operator or a left bracket
if (*pToken == '(')
{
// it may be a left bracket
dealWithLeftBrac(pToken);
lastToken = LEFT_BRAC;
}
else
{
// it may be a number
dealWithNumber(pToken);
lastToken = NUMBER;
}
break;
case RIGHT_BRAC:
// after a right bracket
if (*pToken == ')')
{
// it may be another right bracket
dealWithRightBrac(pToken);
lastToken = RIGHT_BRAC;
}
else
{
// it may be an perator
dealWithOperator(pToken);
lastToken = OPERATOR;
}
break;
}
} while (!_stkOperators.empty())
{
if (_stkOperators.top() == '(')
{
throw string("bad token '('");
}
calculateStack();
} assert(!_stkNodes.empty());
_root = _stkNodes.top();
} void Calculator::postOrder(BinaryTreeNode<Token> *node)
{
if (node)
{
postOrder(node->_leftChild);
postOrder(node->_rightChild);
// visit binary tree data
if (node->_data._type == NUMBER)
{
_stkNumbers.push(node->_data._data.num);
}
else
{
assert(!_stkNumbers.empty());
double d2 = _stkNumbers.top();
_stkNumbers.pop();
assert(!_stkNumbers.empty());
double d1 = _stkNumbers.top();
_stkNumbers.pop();
char op = node->_data._data.op;
_stkNumbers.push(calculate(d1, op, d2));
}
}
} double Calculator::calculate()
{
while (!_stkNumbers.empty())
{
_stkNumbers.pop();
} BinaryTree::postOrder(); assert(!_stkNumbers.empty());
return _stkNumbers.top();
} void Calculator::dealWithNumber(char *&pToken) throw(string)
{
if (!isdigit(*pToken) && *pToken != '-')
{
throw string("bad token '") + *pToken + "'";
} BinaryTreeNode<Token> * node = new BinaryTreeNode<Token>();
assert(node);
node->_data._type = NUMBER;
node->_data._data.num = strtod(pToken, &pToken);
node->_leftChild = node->_rightChild = nullptr;
_stkNodes.push(node);
} void Calculator::dealWithOperator(char *&pToken) throw(string)
{
if (*pToken != '+' && *pToken != '-' && *pToken != '*' && *pToken != '/')
{
throw string("bad token '") + *pToken + "'";
} if (!_stkOperators.empty()
&& priority(_stkOperators.top()) >= priority(*pToken))
{
calculateStack();
}
_stkOperators.push(*pToken++);
} void Calculator::dealWithLeftBrac(char *&pToken) throw(string)
{
if (*pToken != '(')
{
throw string("bad token '") + *pToken + "'";
} _stkOperators.push(*pToken++);
} void Calculator::dealWithRightBrac(char *&pToken) throw(string)
{
if (*pToken != ')')
{
throw string("bad token '") + *pToken + "'";
} while (!_stkOperators.empty() && _stkOperators.top() != '(')
{
calculateStack();
if (_stkOperators.empty())
{
throw string("bad token ')'");
}
}
_stkOperators.pop();
pToken++;
} // main.cpp
#include "calculator.h" int main(int argc, char *argv[])
{
Calculator calculator; if (argc > )
{
if (argc == )
{
calculator.parseExpression(string(argv[]));
cout << calculator.calculate() << endl;
}
else
{
cout << "too many arguments" << endl;
}
}
else
{
while ()
{
string expression;
cout << ">" << flush;
cin >> expression;
if (expression == "quit;")
{
cout << "Bye." << endl;
return ;
}
try
{
calculator.parseExpression(expression);
cout << calculator.calculate() << endl;
}
catch (string ex)
{
cout << ex << endl;
}
}
}
}
表达式求值(二叉树方法/C++语言描述)(四)的更多相关文章
- 表达式求值(二叉树方法/C++语言描述)(二)
表达式二叉树节点的数据可能是运算数或运算符,可以使用一个联合体进行存储:同时还需要一个变量来指示存储的是运算数还是运算符,可以采用和栈方法求值中一样的枚举类型TokenType: typedef en ...
- 表达式求值(二叉树方法/C++语言描述)(一)
使用二叉树对算数表达式(以下简称为表达式)进行求值,实质上是将表达式转换为二叉树,对其进行后序遍历,得到后缀表达式的同时可以求得表达式的值.转换和求值的过程也需要借助数据结构栈的帮助. 二叉树数据结构 ...
- 表达式求值(二叉树方法/C++语言描述)(三)
二叉树方法求值对运算数处理的方法与栈方法求值不太相同,除了将字符串中的运算数转换为浮点类型外,还需要生成新的节点: void Calculator::dealWithNumber(char *& ...
- 表达式求值(二叉树方法/C++语言描述)(五)
本例中的二叉树图是使用Graphviz绘制的(Graphviz官网),在Ubuntu Linux下可以使用apt-get命令安装它: sudo apt-get install graphviz 表达式 ...
- 表达式求值--数据结构C语言算法实现
这篇博客介绍的表达式求值是用C语言实现的,只使用了c++里面的引用. 数据结构课本上的一个例题,但是看起来很简单,实现却遇到了很多问题. 这个题需要构建两个栈,一个用来存储运算符OPTR, 一个用来存 ...
- LeetCode:逆波兰表达式求值【150】
LeetCode:逆波兰表达式求值[150] 题目描述 根据逆波兰表示法,求表达式的值. 有效的运算符包括 +, -, *, / .每个运算对象可以是整数,也可以是另一个逆波兰表达式. 说明: 整数除 ...
- 表达式求值(栈方法/C++语言描述)(二)
上篇中完成了对表达式求值的整体过程,接下来看看如何处理不同类型的token. 对运算数的处理比较简单,它直接调用函数strtod(),将字符串中的运算数转换为浮点类型并将它压入运算数栈中: void ...
- 利用栈实现算术表达式求值(Java语言描述)
利用栈实现算术表达式求值(Java语言描述) 算术表达式求值是栈的典型应用,自己写栈,实现Java栈算术表达式求值,涉及栈,编译原理方面的知识.声明:部分代码参考自茫茫大海的专栏. 链栈的实现: pa ...
- Java描述表达式求值的两种解法:双栈结构和二叉树
Java描述表达式求值的两种解法:双栈结构和二叉树 原题大意:表达式求值 求一个非负整数四则混合运算且含嵌套括号表达式的值.如: # 输入: 1+2*(6/2)-4 # 输出: 3.0 数据保证: 保 ...
随机推荐
- phpcms v9栏目列表调用每一篇文章内容方法
{pc:content action="lists" catid="$catid" num="25" order="id DESC ...
- 通过 itms:services://? 在线安装ipa ,跨过app-store
1.需要一个html文件,引导下载用户在线安装ipa <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN&quo ...
- Python之数据加密与解密及相关操作(hashlib模块、hmac模块、random模块、base64模块、pycrypto模块)
本文内容 数据加密概述 Python中实现数据加密的模块简介 hashlib与hmac模块介绍 random与secrets模块介绍 base64模块介绍 pycrypto模块介绍 总结 参考文档 提 ...
- SSH连接不上CentOS 主机配置文件导致的原因的解决方法
一.CentOS之SSH的安装与配置 SSH 为 Secure Shell 的缩写,由 IETF 的网络工作小组(Network Working Group)所制定SSH 为建立在应用层和传输层基础上 ...
- java中byte, iso-8859-1, UTF-8,乱码的根源
Post@https://ryan-miao.github.io 背景 还是多语言, 在项目中遇到本地环境和服务端环境不一致乱码的情形.因此需要搞清楚乱码产生的过程,来分析原因. 获取多语言代码如下: ...
- .net core web api + Autofac + EFCore 个人实践
1.背景 去年时候,写过一篇<Vue2.0 + Element-UI + WebAPI实践:简易个人记账系统>,采用Asp.net Web API + Element-UI.当时主要是为了 ...
- phpstorm显示行号
在Windows上. 其View->Active Editor->Show Line Numbers (仅适用于当前和变化File->Settings->Editor-> ...
- 微信支付(APP)
折腾了一天,终于搞定了微信支付,总结一下.首先从服务器端获取prepareid,Andorid 端再根据这个prepareid二次签名. 服务器端: 从官网上下载DEMO,Demo中只有JsAPi,M ...
- Ionic开发笔记
Ionic 开发笔记 记录开发中遇到的一些问题 ion-side-menu,使所有顶部导航标题居中 <!-- 添加 align-title="center" 使顶部导航标题居 ...
- lvs讲解
作者写的就是好呀,简单明了,安逸的我也不知道早点看看,非得等到找工作了,在这里临时抱佛脚,以后再过来添加我自己的总结 http://www.linuxvirtualserver.org/zh/