表达式求值(二叉树方法/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 数据保证: 保 ...
随机推荐
- Git配合Tag的代码回滚
现有的远程仓库版本的tag为v1.0 前置准备 具体操作: 我们在本地修改一下readme文件,然后进行add,commit操作. 再给我们的commit打上tag git tag -a v1.1 - ...
- Entity Framework入门教程:创建实体数据模型
下图为一个已经创建好的数据库表关系 实体数据模型的创建过程 在Visual Studio项目中,右键程序集菜单,选择[添加]->[新建项],在[添加新项窗口]中选择[ADO.NET实体数据模型] ...
- jq-animate实现返回顶部效果
jq-animate实现返回顶部效果: <!doctype html> <html lang="en"> <head> <meta cha ...
- java对mysql的增删改查
-----连接数据库 package connectdb;import java.sql.*;class Dbcon { // 此处连接数据库,独立开一个类,以后操作数据库的每次连接就不用写这么多 p ...
- 使用Scribefire在博客中插入语法高亮 II
效果如下, 这是我们在Scribefire中添加的code按钮,单击此按钮,则会出现 在codeHere中直接输入代码就可以了. 查看html 可以看到,其中已经添加了<pre>标签. 下 ...
- H5仿微信界面教程(一)
前言 先来张图,仿微信界面,界面如下,并不完全一模一样,只能说有些类似,希望大家见谅. 1 用到的知识点 jQuery WeUI 是WeUI的一个jQuery实现版本,除了实现了官方插件之外,它还提供 ...
- Bootstrap提示信息(标签、徽章、巨幕和页头)
前面的话 在Bootstrap中,有一些组件用于提示信息,如 标签.徽章.巨幕和页头.本文将详细介绍Bootstrap提示信息 标签 在一些Web页面中常常会添加一个标签用来告诉用户一些额外的信息,比 ...
- Python3 常用数据类型语法
1.int类型 int类型的数据是没有长度限制的,它的最大长度只与计算机的内存有关. bin(i) 返回二进制表示结果, hex(i) 十六进制, int(i) 整数( ...
- Excel文件按照指定模板导入数据(用jxl.jar包)
本文中的方法只适合Excel2003,要读取Excel2007最好使用poi.jar,据说poi.jar还在更新,jxl.jar已经不更新了,处理Excel文件的读写问题最好还是学习poi.j ...
- jquery.validata.js 插件2
结合上面的,今天写一下validate的使用方法. validate()验证所选的form. validate 方法返回一个 Validator 对象.Validator 对象有很多方法可以用来引发校 ...