Java實(shí)現(xiàn)四則混合運(yùn)算代碼示例
使用棧來(lái)實(shí)現(xiàn),可以處理運(yùn)算優(yōu)先級(jí)。
使用自然四則運(yùn)算表達(dá)式即可,如:4+(3*(3-1)+2)/2。無(wú)需把表達(dá)式先轉(zhuǎn)換為逆波蘭等形式。
package com.joshua.cal; import java.util.Collections; import java.util.HashMap; import java.util.HashSet; import java.util.LinkedList; import java.util.List; import java.util.Map; import java.util.Set; import java.util.Stack; public class Calculator { private final Stack<Double> numStack = new Stack<Double>(); private final Stack<Character> opStack = new Stack<Character>(); private char currentOperator; private char opStackTop; private int i; private String expression; @SuppressWarnings("rawtypes") public void exec(String expression) { try { clean(); if (expression == null || expression.isEmpty()) { throw new IllegalArgumentException("Blank Expression!"); } this.expression = expression; opStack.push(TERMINATE_TOKENS.START_END_MARK); List tokens = TOKENIZER.exec(expression + TERMINATE_TOKENS.START_END_MARK); for (; i < tokens.size(); i++) { final Object token = tokens.get(i); if (token instanceof Double) { processOperand((double) token); } else { processOperator((char) token); } } } catch (Throwable e) { System.err.println(String.format( "Incorret Expression: %s\nError: %s", expression, e.getMessage())); } } private void processOperand(final double operand) { numStack.push(operand); } private void processOperator(final char currentOperator) { this.currentOperator = currentOperator; this.opStackTop = opStack.peek(); char calMode = CALCULATE_MODE.getRule(currentOperator, opStackTop); switch (calMode) { case '>': processStackHigerPriorityOperator(); break; case '<': processStackLowerPriorityOperator(); break; case '=': processStackEqualPriorityOperator(); break; default: break; } } private void processStackLowerPriorityOperator() { opStack.push(currentOperator); } private void processStackHigerPriorityOperator() { numStack.push(CALCULATE.exec(opStack.pop(), numStack.pop(), numStack.pop())); --i; // pointer back to the previous operator. } private void processStackEqualPriorityOperator() { if (TERMINATE_TOKENS.START_END_MARK == currentOperator) { System.out.println(expression + " = " + numStack.peek()); } else if (')' == currentOperator) { opStack.pop(); } } public void clean() { numStack.clear(); opStack.clear(); i = 0; } public static void main(String[] args) { Calculator cal = new Calculator(); cal.exec("4+(3*(3-1)+2)/2"); // = 8 cal.exec("4 + (-3 * ( 3 - 1 ) + 2)"); // = 0 cal.exec("4 +-/ (-3 * ( 3 - 1 ) + 2)"); // incorrect expression! cal.exec("4.5+(3.2+3)/2"); // = 7.6 cal.exec("4.5+(3.2:3)/2"); // incorrect expression! cal.exec("-4.5+(3.2-3)/2"); // = -4.4 } } enum CALCULATE { INSTANCE; public static double exec(final char operator, final double right, final double left) { switch (operator) { case '+': return left + right; case '-': return left - right; case '*': return left * right; case '/': return left / right; default: throw new IllegalArgumentException("Unsupported operator: " + operator); } } } enum TERMINATE_TOKENS { INSTANCE; public static final char START_END_MARK = '#'; private static final Map<Character, Integer> TOKENs = new HashMap<Character, Integer>(); static { // token, token id TOKENs.put('+', 0); TOKENs.put('-', 1); TOKENs.put('*', 2); TOKENs.put('/', 3); TOKENs.put('(', 4); TOKENs.put(')', 5); TOKENs.put(START_END_MARK, 6); } private static Set<Character> NEGATIVE_NUM_SENSITIVE = new HashSet<Character>(); public static synchronized Set<Character> getNegativeNumSensitiveToken() { if (NEGATIVE_NUM_SENSITIVE.size() == 0) { NEGATIVE_NUM_SENSITIVE.addAll(TOKENs.keySet()); NEGATIVE_NUM_SENSITIVE.remove(')'); } return NEGATIVE_NUM_SENSITIVE; } public static boolean isTerminateToken(final char token) { Set<Character> keys = TOKENs.keySet(); return keys.contains(token); } public static int getTokenId(final char token) { return TOKENs.get(token) == null ? -1 : TOKENs.get(token); } public static int getTokenSize() { return TOKENs.size(); } } enum CALCULATE_MODE { INSTANCE; private static char[][] RULES = { // + - * / ( ) # { '>', '>', '<', '<', '<', '>', '>' }, // + { '>', '>', '<', '<', '<', '>', '>' }, // - { '>', '>', '>', '>', '<', '>', '>' }, // * { '>', '>', '>', '>', '<', '>', '>' }, // / { '<', '<', '<', '<', '<', '=', 'o' }, // ( { '>', '>', '>', '>', 'o', '>', '>' }, // ) { '<', '<', '<', '<', '<', 'o', '=' }, // # }; static { if (RULES.length != TERMINATE_TOKENS.getTokenSize() || RULES.length < 1 || RULES[0].length != TERMINATE_TOKENS.getTokenSize()) { throw new IllegalArgumentException("Rules matrix is incorrect!"); } } public static char getRule(final char currentOperator, final char opStackTop) { try { return RULES[TERMINATE_TOKENS.getTokenId(opStackTop)][TERMINATE_TOKENS .getTokenId(currentOperator)]; } catch (Throwable e) { throw new RuntimeException("No rules were defined for some token!"); } } } enum TOKENIZER { INSTANCE; private static final StringBuilder BUFFER = new StringBuilder(); private static String clearExpression(String expression) { return expression.replaceAll(" ", ""); } private static Character PREVIOUS_CHAR; private static void clean() { BUFFER.delete(0, BUFFER.length()); PREVIOUS_CHAR = null; } private static boolean processNegativeNumbers(final String exp, final int index) { char c = exp.charAt(index); if (('+' == c || '-' == c) && (PREVIOUS_CHAR == null || TERMINATE_TOKENS .getNegativeNumSensitiveToken().contains(PREVIOUS_CHAR)) && !TERMINATE_TOKENS.isTerminateToken(exp.charAt(index + 1))) { BUFFER.append(c); return true; } return false; } @SuppressWarnings({ "unchecked", "rawtypes" }) public static List<?> exec(final String expression) { clean(); String exp = clearExpression(expression); List result = new LinkedList(); for (int i = 0; i < exp.length(); i++) { char c = exp.charAt(i); if (TERMINATE_TOKENS.isTerminateToken(c)) { if (processNegativeNumbers(exp, i)) continue; if (BUFFER.length() > 0) { result.add(Double.valueOf(BUFFER.toString())); BUFFER.delete(0, BUFFER.length()); } result.add(c); } else { BUFFER.append(c); } PREVIOUS_CHAR = c; } return Collections.unmodifiableList(result); } }
輸出
4+(3*(3-1)+2)/2 = 8.0 4 + (-3 * ( 3 - 1 ) + 2) = 0.0 4.5+(3.2+3)/2 = 7.6 -4.5+(3.2-3)/2 = -4.4 Incorret Expression: 4 +-/ (-3 * ( 3 - 1 ) + 2) Error: null Incorret Expression: 4.5+(3.2:3)/2 Error: For input string: "3.2:3"
總結(jié)
以上就是本文關(guān)于Java實(shí)現(xiàn)四則混合運(yùn)算代碼示例的全部?jī)?nèi)容,希望對(duì)大家有所幫助。感興趣的朋友可以參閱:大話Java混合運(yùn)算規(guī)則 淺談Java變量賦值運(yùn)算符及相關(guān)實(shí)例 Java大數(shù)字運(yùn)算之BigInteger 等,有什么問(wèn)題可以隨時(shí)留言,小編會(huì)及時(shí)回復(fù)大家的。感謝朋友們對(duì)腳本之家網(wǎng)站的支持。
相關(guān)文章
Mybatis-Plus同時(shí)使用邏輯刪除和唯一索引的問(wèn)題及解決辦法(報(bào)數(shù)據(jù)重復(fù)Duplicate entry的
在開(kāi)發(fā)中,我們經(jīng)常會(huì)有邏輯刪除和唯一索引同時(shí)使用的情況,但當(dāng)使用mybatis plus時(shí),如果同時(shí)使用邏輯刪除和唯一索引,會(huì)報(bào)數(shù)據(jù)重復(fù)Duplicate entry的問(wèn)題,如何解決這個(gè)問(wèn)題呢,小編給大家分享Mybatis-Plus同時(shí)使用邏輯刪除和唯一索引的問(wèn)題及解決辦法,一起看看吧2023-11-11Mabitis中的#與$符號(hào)區(qū)別及用法介紹
這篇文章主要介紹了Mabitis中的#與$符號(hào)區(qū)別,需要的朋友可以參考下2017-02-02Java mutable對(duì)象和immutable對(duì)象的區(qū)別說(shuō)明
這篇文章主要介紹了Java mutable對(duì)象和immutable對(duì)象的區(qū)別,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-06-06詳解Spring Cloud Hystrix斷路器實(shí)現(xiàn)容錯(cuò)和降級(jí)
本篇文章主要介紹了詳解Spring Cloud Hystrix斷路器實(shí)現(xiàn)容錯(cuò)和降級(jí),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-05-05Java Swing中的JButton、JComboBox、JList和JColorChooser組件使用案例
這篇文章主要介紹了Java Swing中的按鈕(JButton)、組合框(JComboBox)、下拉列表(JList)和顏色選擇器(JColorChooser)組件使用案例,需要的朋友可以參考下2014-10-10如何解決LocalDateTime傳值JSON格式化問(wèn)題
這篇文章主要介紹了如何解決LocalDateTime傳值JSON格式化問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-08-08