计算一个表达式的值
2007-06-11 23:01
405 查看
中缀表达式 --> 后缀表达式,转换过程中同时计算
操作符堆栈(OperatorHeap):
import java.util.Arrays;
public class OperatorHeap
...{
private char[] data;
private int size;
public OperatorHeap()
...{
this(10);
}
public OperatorHeap(int size)
...{
data = new char[size];
initHeap();
}
public void initHeap()
...{
data[size++] = '#';
}
//栈顶
public char top()
...{
return data[size-1];
}
//入栈
public void push(char c)
...{
ensureCapacity(size + 1);
data[size++] = c;
}
//出栈
public char pop()
...{
return data[--size];
}
//扩展容量
public void ensureCapacity(int minSize)
...{
int oldCapacity = data.length;
if (minSize > oldCapacity) ...{
int newCapacity = (oldCapacity * 3 / 2 + 1) > minSize ?
oldCapacity * 3 / 2 + 1 : minSize;
data = (char[]) Arrays.copyOf(data, newCapacity);
}
}
}
操作数堆栈(NumHeap):
import java.util.Arrays;
public class NumHeap
...{
private double[] data;
private int size;
public NumHeap()
...{
this(10);
}
public NumHeap(int size)
15462
...{
data = new double[size];
}
//栈顶
public double top()
...{
return data[size-1];
}
//入栈
public void push(double c)
...{
ensureCapacity(size + 1);
data[size++] = c;
}
//出栈
public double pop()
...{
return data[--size];
}
//扩展容量
public void ensureCapacity(int minSize)
...{
int oldCapacity = data.length;
if (minSize > oldCapacity) ...{
int newCapacity = (oldCapacity * 3 / 2 + 1) > minSize ?
oldCapacity * 3 / 2 + 1 : minSize;
data = (double[]) Arrays.copyOf(data, newCapacity);
}
}
}
Eval类,提供了一个静态的eval方法,用于计算表达式值:
/** *//**
* 计算表达式的值,如 3+(2+9*7)/5
* @author Liw
* @time 2007-6
*/
public class Eval
...{
public static double eval(String str)
...{
str = str + "#"; //设置"#"优先级最低,保证opHeap里的运算符能够全部退栈
String[] words = str.split("(?<!^-?|[+/*()-]-)((?<=[+/*()-])|(?=[+/*()-]))"); //分割表达式
OperatorHeap opHeap = new OperatorHeap(); //操作符堆栈
NumHeap numHeap = new NumHeap(); //操作数堆栈
for (int i = 0; i < words.length; i++) ...{
if (words[i].matches("-?//d+|-?//d+//.//d+")) ...{ //操作数
numHeap.push(Double.parseDouble(words[i])); //操作数入栈
}
else ...{ //操作符
char op_top = opHeap.top(), op_cur = words[i].toCharArray()[0]; //栈顶操作符&当前读取的操作符
while (compare(op_top, op_cur)) ...{ //若当前操作符op_cur优先级低于栈顶,退栈,直到op_cur优先级大于栈顶
if ((op_top == '(' && op_cur == ')') || (op_top == '#' && op_cur == '#')) ...{
opHeap.pop(); //弹出"(" or "#"
break;
}
else ...{ //弹出一运算符
double a = numHeap.pop(), b = numHeap.pop(); //弹出两个数字用于计算
numHeap.push(eval2(b, a, opHeap.pop())); //弹出一运算符,计算两个数的值,结果入栈
op_top = opHeap.top(); //新的栈顶
}
}
if (op_cur != ')') ...{
opHeap.push(op_cur); //操作符op_cur入栈
}
}
}
return numHeap.top();
}
private static double eval2(double a, double b, char op)
...{
switch (op)
...{
case '+':
return a + b;
case '-':
return a - b;
case '*':
return a * b;
case '/':
return a / b;
default:
return -1;
}
}
//返回两个运算符之间的优先级,若op1>op2,返回true
private static boolean compare(char op1, char op2)
...{
return !(op1 == '(' && op2 == '(') //连续两括号 <-- false
&& (getOpLevel(op1) == getOpLevel(op2)) //相同运算符 <-- true
|| !(op1 == '(' || op2 == '(' || op1 == '#') //括号 <-- false
&& getOpLevel(op1) > getOpLevel(op2); //普通优先级比较
}
//忽略括号时的优先级
private static int getOpLevel(char op)
...{
switch(op)
...{
case '+':
case '-':
return 1;
case '*':
case '/':
return 2;
case '(':
case ')':
return 0;
default:
return -1; //"#"
}
}
}
如下使用:
public class Test
...{
public static void main(String[] args)
...{
System.out.println(Eval.eval("3+(2+9*7)/5"));
}
}
操作符堆栈(OperatorHeap):
import java.util.Arrays;
public class OperatorHeap
...{
private char[] data;
private int size;
public OperatorHeap()
...{
this(10);
}
public OperatorHeap(int size)
...{
data = new char[size];
initHeap();
}
public void initHeap()
...{
data[size++] = '#';
}
//栈顶
public char top()
...{
return data[size-1];
}
//入栈
public void push(char c)
...{
ensureCapacity(size + 1);
data[size++] = c;
}
//出栈
public char pop()
...{
return data[--size];
}
//扩展容量
public void ensureCapacity(int minSize)
...{
int oldCapacity = data.length;
if (minSize > oldCapacity) ...{
int newCapacity = (oldCapacity * 3 / 2 + 1) > minSize ?
oldCapacity * 3 / 2 + 1 : minSize;
data = (char[]) Arrays.copyOf(data, newCapacity);
}
}
}
操作数堆栈(NumHeap):
import java.util.Arrays;
public class NumHeap
...{
private double[] data;
private int size;
public NumHeap()
...{
this(10);
}
public NumHeap(int size)
15462
...{
data = new double[size];
}
//栈顶
public double top()
...{
return data[size-1];
}
//入栈
public void push(double c)
...{
ensureCapacity(size + 1);
data[size++] = c;
}
//出栈
public double pop()
...{
return data[--size];
}
//扩展容量
public void ensureCapacity(int minSize)
...{
int oldCapacity = data.length;
if (minSize > oldCapacity) ...{
int newCapacity = (oldCapacity * 3 / 2 + 1) > minSize ?
oldCapacity * 3 / 2 + 1 : minSize;
data = (double[]) Arrays.copyOf(data, newCapacity);
}
}
}
Eval类,提供了一个静态的eval方法,用于计算表达式值:
/** *//**
* 计算表达式的值,如 3+(2+9*7)/5
* @author Liw
* @time 2007-6
*/
public class Eval
...{
public static double eval(String str)
...{
str = str + "#"; //设置"#"优先级最低,保证opHeap里的运算符能够全部退栈
String[] words = str.split("(?<!^-?|[+/*()-]-)((?<=[+/*()-])|(?=[+/*()-]))"); //分割表达式
OperatorHeap opHeap = new OperatorHeap(); //操作符堆栈
NumHeap numHeap = new NumHeap(); //操作数堆栈
for (int i = 0; i < words.length; i++) ...{
if (words[i].matches("-?//d+|-?//d+//.//d+")) ...{ //操作数
numHeap.push(Double.parseDouble(words[i])); //操作数入栈
}
else ...{ //操作符
char op_top = opHeap.top(), op_cur = words[i].toCharArray()[0]; //栈顶操作符&当前读取的操作符
while (compare(op_top, op_cur)) ...{ //若当前操作符op_cur优先级低于栈顶,退栈,直到op_cur优先级大于栈顶
if ((op_top == '(' && op_cur == ')') || (op_top == '#' && op_cur == '#')) ...{
opHeap.pop(); //弹出"(" or "#"
break;
}
else ...{ //弹出一运算符
double a = numHeap.pop(), b = numHeap.pop(); //弹出两个数字用于计算
numHeap.push(eval2(b, a, opHeap.pop())); //弹出一运算符,计算两个数的值,结果入栈
op_top = opHeap.top(); //新的栈顶
}
}
if (op_cur != ')') ...{
opHeap.push(op_cur); //操作符op_cur入栈
}
}
}
return numHeap.top();
}
private static double eval2(double a, double b, char op)
...{
switch (op)
...{
case '+':
return a + b;
case '-':
return a - b;
case '*':
return a * b;
case '/':
return a / b;
default:
return -1;
}
}
//返回两个运算符之间的优先级,若op1>op2,返回true
private static boolean compare(char op1, char op2)
...{
return !(op1 == '(' && op2 == '(') //连续两括号 <-- false
&& (getOpLevel(op1) == getOpLevel(op2)) //相同运算符 <-- true
|| !(op1 == '(' || op2 == '(' || op1 == '#') //括号 <-- false
&& getOpLevel(op1) > getOpLevel(op2); //普通优先级比较
}
//忽略括号时的优先级
private static int getOpLevel(char op)
...{
switch(op)
...{
case '+':
case '-':
return 1;
case '*':
case '/':
return 2;
case '(':
case ')':
return 0;
default:
return -1; //"#"
}
}
}
如下使用:
public class Test
...{
public static void main(String[] args)
...{
System.out.println(Eval.eval("3+(2+9*7)/5"));
}
}
相关文章推荐
- 利用表达式树,计算一个加减乘除(可带括号)的表达式
- 我就给一个PHP逆波兰表达式的算法吧---工资计算专用
- 一个表达式计算案例的设计和实现!〈转贴〉
- LEX & YACC 写的一个表达式计算程序
- 计算任意一个数学运算表达式字符串的值
- 输入一个只包含个位数字的简单四则运算表达式字符串,计算该表达式的值
- 计算一个表达式的值
- [Happy Coding] 一个正则表达式,支持逻辑和关系运算符组成的表达式计算
- 我就给一个PHP逆波兰表达式的算法吧---工资计算专用
- 给定一个表示分数加减表达式的字符串,需要以字符串格式返回计算结果。结果表达式也是分数式
- 华为机试:计算一个表达式的值
- 我就给一个PHP逆波兰表达式的算法吧---工资计算专用
- 输入一个表达式,没有括号,数字小于0-9之间,输出计算结果,所有的中间结果化为整形。 例如: 输入:3+8×2/9-2 输出:2
- 提供一个在线表达式计算服务
- 一个表达式计算案例的设计和实现
- 我就给一个PHP逆波兰表达式的算法吧---工资计算专用
- 我就给一个PHP逆波兰表达式的算法吧---工资计算专用
- 读入一个只包含 +, -, *, / 的非负整数计算表达式,计算该表达式的值。
- 一个基于LL(1)的简易C++四则表达式计算
- 输入一个字符串表达式,输出计算结果