java實現任意四則運算表達式求值算法
本文實例講述了java實現任意四則運算表達式求值算法。分享給大家供大家參考。具體分析如下:
該程序用于計算任意四則運算表達式。如 4 * ( 10 + 2 ) + 1 的結果應該為 49。
算法說明:
1. 首先定義運算符優(yōu)先級。我們用一個
Map<String, Map<String, String>>
來保存優(yōu)先級表。這樣我們就可以通過下面的方式來計算兩個運算符的優(yōu)先級了:
/**
 * 查表得到op1和op2的優(yōu)先級
 * @param op1 運算符1
 * @param op2 運算符2
 * @return ">", "<" 或 "="
 */
public String priority(String op1, String op2) {
 return priorityMap.get(op1).get(op2);
}
2. 掃描表達式字符串,每次讀入一個 token 進行處理。
使用兩個輔助棧:optStack用于保存運算符,numStack用于保存操作數. 我們用 '#' 作為表達式的起始和結果標志符。
讀入一個token,如果它是數字,則壓入numStack棧中;
如果它是運算符,則取出optStack棧的棧頂元素A,將 A 與 token 進行優(yōu)先級比較。
如果 A < token,則將 token 壓入optStack棧。
如果 A = token,則說明 token和A是一對括號,此時將optStack棧的棧頂元素彈出。
如果 A > token,則從numStack中彈出2個操作數,從optStack中彈出1個運算符,并計算結果。
當optStrack棧為空時(即棧頂元素為 '#'),numStack棧的棧頂元素即為表達式的值。
算法實現:
/**
 * 算術表達式求值。
 * 3 + 4 * 12 結果為51
 * @author whf
 *
 */
public class EvaluateExpression {
 // 運算符優(yōu)先級關系表
 private Map<String, Map<String, String>> priorityMap = new HashMap<String, Map<String, String>>();
 private LinkedStack<String> optStack = new LinkedStack<String>();
 // 運算符棧
 private LinkedStack<Double> numStack = new LinkedStack<Double>();
 // 操作數棧
 /**
  * 計算表達式
  * @param exp 四則運算表達式, 每個符號必須以空格分隔
  * @return
  */
 public double calcualte(String exp) {
  StringTokenizer st = new StringTokenizer(exp);
  while (st.hasMoreTokens()) {
   String token = st.nextToken();
   process(token);
  }
  return numStack.pop();
 }
 /**
  * 讀入一個符號串。
  * 如果是數字,則壓入numStack
  * 如果是運算符,則將optStack棧頂元素與該運算符進行優(yōu)先級比較
  * 如果棧頂元素優(yōu)先級低,則將運算符壓入optStack棧,如果相同,則彈出左括號,如果高,則取出2個數字,取出一個運算符執(zhí)行計算,然后將結果壓入numStack棧中
  * @param token
  */
 private void process(String token) {
  while (false == "#".equals(optStack.getTop()) || false == token.equals("#")) {
   // token is numeric
   if (true == isNumber(token)) {
        numStack.push(Double.parseDouble(token));
        break;
        // token is operator
   } else {
        String priority = priority(optStack.getTop(), token);
        if ("<".equals(priority)) {
         optStack.push(token);
         break;
        } else if ("=".equals(priority)) {
         optStack.pop();
         break;
        } else {
          double res = calculate(optStack.pop(), numStack.pop(), numStack.pop());
          numStack.push(res);
        }
   }
  }
 }
 /**
  * 執(zhí)行四則運算
  * @param opt
  * @param n1
  * @param n2
  * @return
  */
 private double calculate(String opt, double n1, double n2) {
  if ("+".equals(opt)) {
   return n2 + n1;
  } else if ("-".equals(opt)) {
   return n2 - n1;
  } else if ("*".equals(opt)) {
   return n2 * n1;
  } else if ("/".equals(opt)) {
   return n2 / n1;
  } else {
   throw new RuntimeException("unsupported operator:" + opt);
  }
 }
 /**
  * 檢查一個String是否為數字
  * @param token
  * @return
  */
 private boolean isNumber(String token) {
  int LEN = token.length();
  for (int ix = 0 ; ix < LEN ; ++ix) {
   char ch = token.charAt(ix);
   // 跳過小數點
   if (ch == '.') {
    continue;
   }
   if (false == isNumber(ch)) {
    return false;
   }
  }
  return true;
 }
 /**
  * 檢查一個字符是否為數字
  * @param ch
  * @return
  */
 private boolean isNumber(char ch) {
  if (ch >= '0' && ch <= '9') {
   return true;
  }
  return false;
 }
 /**
  * 查表得到op1和op2的優(yōu)先級
  * @param op1 運算符1
  * @param op2 運算符2
  * @return ">", "<" 或 "="
  */
 public String priority(String op1, String op2) {
  return priorityMap.get(op1).get(op2);
 }
 /**
  * 構造方法,初始化優(yōu)先級表
  */
 public EvaluateExpression() {
  // initialize stack
  optStack.push("#");
  // initialize priority table
  // +
  Map<String, String> subMap = new HashMap<String, String>();
  subMap.put("+", ">");
  subMap.put("-", ">");
  subMap.put("*", "<");
  subMap.put("/", "<");
  subMap.put("(", "<");
  subMap.put(")", ">");
  subMap.put("#", ">");
  priorityMap.put("+", subMap);
  // -
  subMap = new HashMap<String, String>();
  subMap.put("+", ">");
  subMap.put("-", ">");
  subMap.put("*", "<");
  subMap.put("/", "<");
  subMap.put("(", "<");
  subMap.put(")", ">");
  subMap.put("#", ">");
  priorityMap.put("-", subMap);
  // *
  subMap = new HashMap<String, String>();
  subMap.put("+", ">");
  subMap.put("-", ">");
  subMap.put("*", ">");
  subMap.put("/", ">");
  subMap.put("(", "<");
  subMap.put(")", ">");
  subMap.put("#", ">");
  priorityMap.put("*", subMap);
  // /
  subMap = new HashMap<String, String>();
  subMap.put("+", ">");
  subMap.put("-", ">");
  subMap.put("*", ">");
  subMap.put("/", ">");
  subMap.put("(", "<");
  subMap.put(")", ">");
  subMap.put("#", ">");
  priorityMap.put("/", subMap);
  // (
  subMap = new HashMap<String, String>();
  subMap.put("+", "<");
  subMap.put("-", "<");
  subMap.put("*", "<");
  subMap.put("/", "<");
  subMap.put("(", "<");
  subMap.put(")", "=");
  //subMap.put("#", ">");
  priorityMap.put("(", subMap);
  // )
  subMap = new HashMap<String, String>();
  subMap.put("+", ">");
  subMap.put("-", ">");
  subMap.put("*", ">");
  subMap.put("/", ">");
  //subMap.put("(", "<");
  subMap.put(")", ">");
  subMap.put("#", ">");
  priorityMap.put(")", subMap);
  // #
  subMap = new HashMap<String, String>();
  subMap.put("+", "<");
  subMap.put("-", "<");
  subMap.put("*", "<");
  subMap.put("/", "<");
  subMap.put("(", "<");
  //subMap.put(")", ">");
  subMap.put("#", "=");
  priorityMap.put("#", subMap);
 }
}
程序測試:
String exp = "4 * ( 10 + 2 ) + 1 #"; EvaluateExpression ee = new EvaluateExpression(); out.println(ee.calcualte(exp));
運行結果為 49。
希望本文所述對大家的C++程序設計有所幫助。
上一篇:C++語言實現線性表之數組實例
欄 目:C語言
本文標題:java實現任意四則運算表達式求值算法
本文地址:http://www.jygsgssxh.com/a1/Cyuyan/3090.html
您可能感興趣的文章
- 01-10數據結構課程設計-用棧實現表達式求值的方法詳解
 - 01-10使用OpenGL實現3D立體顯示的程序代碼
 - 01-10求斐波那契(Fibonacci)數列通項的七種實現方法
 - 01-10C語言 解決不用+、-、&#215;、&#247;數字運算符做加法
 - 01-10使用C++實現全排列算法的方法詳解
 - 01-10用C++實現DBSCAN聚類算法
 - 01-10深入全排列算法及其實現方法
 - 01-10全排列算法的非遞歸實現與遞歸實現的方法(C++)
 - 01-10用C語言實現單鏈表的各種操作(一)
 - 01-10用C語言實現單鏈表的各種操作(二)
 


閱讀排行
本欄相關
- 04-02c語言函數調用后清空內存 c語言調用
 - 04-02func函數+在C語言 func函數在c語言中
 - 04-02c語言的正則匹配函數 c語言正則表達
 - 04-02c語言用函數寫分段 用c語言表示分段
 - 04-02c語言中對數函數的表達式 c語言中對
 - 04-02c語言編寫函數冒泡排序 c語言冒泡排
 - 04-02c語言沒有round函數 round c語言
 - 04-02c語言分段函數怎么求 用c語言求分段
 - 04-02C語言中怎么打出三角函數 c語言中怎
 - 04-02c語言調用函數求fibo C語言調用函數求
 
隨機閱讀
- 04-02jquery與jsp,用jquery
 - 08-05dedecms(織夢)副欄目數量限制代碼修改
 - 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
 - 01-10C#中split用法實例總結
 - 01-10delphi制作wav文件的方法
 - 01-10使用C語言求解撲克牌的順子及n個骰子
 - 01-11ajax實現頁面的局部加載
 - 08-05織夢dedecms什么時候用欄目交叉功能?
 - 08-05DEDE織夢data目錄下的sessions文件夾有什
 - 01-10SublimeText編譯C開發(fā)環(huán)境設置
 


