亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關于我們
? 蟲蟲下載站

?? expressionparser.java

?? 學習java的基礎
?? JAVA
字號:
package book.string;

/**
 * 一個簡單的表達式解析器,這個解析器可以計算由數字、運算符和括號組成的表達式的值,并能處理變量,
 * 為了處理簡單,本解析器只支持一個字母的變量,不區分變量字母的大小寫。因此,最多只能存儲26個變量。
 * 如果用戶的變量名長度大于一個字母,則只取第一個字母當作變量。
 */
public class ExpressionParser {

	// 4種標記類型
	/**	標記為空或者結束符	*/
	public static final int NONE_TOKEN = 0;
	/**	標記為分隔符*/
	public static final int DELIMITER_TOKEN = 1;
	/**	標記為變量*/
	public static final int VARIABLE_TOKEN = 2;
	/**	標記為數字*/
	public static final int NUMBER_TOKEN = 3;

	// 4種錯誤類型
	/**	語法錯誤	*/
	public static final int SYNTAX_ERROR = 0;
	/**	括號沒有結束錯誤	*/
	public static final int UNBALPARENS_ERROR = 1;
	/**	表達式為空錯誤	*/
	public static final int NOEXP_ERROR = 2;
	/**	被0除錯誤	*/
	public static final int DIVBYZERO_ERROR = 3;
	//針對四種錯誤類型,定義的四個錯誤提示
	public static final String[] ERROR_MESSAGES = { "Syntax Error", "Unbalanced Parentheses",
			"No Expression Present", "Division by Zero" };
	/**	表達式的結束標記*/
	public static final String EOE = "\0";
	
	/**	表達式字符串*/
	private String exp;  
	/**	解析器當前指針在表達式中的位置*/
	private int expIndex;   
	/**	解析器當前處理的標記*/
	private String token;  
	/**	解析器當前處理標記的類型*/
	private int tokenType; 

	/**	變量的數組*/  
	private double vars[] = new double[26];

	/**
	 * 解析一個表達式,返回表達式的值。
	 * @param expStr	表達式字符串
	 * @return
	 * @throws Exception
	 */  
	public double evaluate(String expStr) throws Exception {
		double result;
		this.exp = expStr;
		this.expIndex = 0;

		//獲取第一個標記
		this.getToken();
		if (this.token.equals(EOE)){
			//沒有表達式異常
			this.handleError(NOEXP_ERROR); // no expression present
		}

		//處理賦值語句
		result = this.parseAssign();
		//處理完賦值語句,應該就是表達式結束符,如果不是,則返回異常
		if (!this.token.equals(EOE)){  
			this.handleError(SYNTAX_ERROR);
		}

		return result;
	}

	/**
	 * 處理賦值語句
	 */  
	private double parseAssign() throws Exception {
		double result;//結果
		int varIndex;//變量下標
		String oldToken;//舊標記
		int oldTokenType;//舊標記的類型

		//如果標記類型是變量
		if (this.tokenType == VARIABLE_TOKEN) {
			// 保存當前標記  
			oldToken = new String(this.token);
			oldTokenType = this.tokenType;
			// 取得變量的索引,本解析器只支持一個字目的變量,
			//如果用戶的變量字母長度大于1,則取第一個字母當作變量  
			varIndex = Character.toUpperCase(this.token.charAt(0)) - 'A';
			
			//獲得下一個標記
			this.getToken();
			//如果當前標記不是等號=
			if (!this.token.equals("=")) {
				//回滾
				this.putBack();  
				// 不是一個賦值語句,將標記恢復到上一個標記  
				this.token = new String(oldToken);
				this.tokenType = oldTokenType;
			} else {
				//如果當前標記是等號=,即給變量賦值,形式如a = 3 + 5;
				//則計算等號后面表達式的值,然后將得到的值賦給變量
				this.getToken();
				//因為加減法的優先級最低,所以計算加減法表達式。
				result = this.parseAddOrSub();
				//將表達式的值賦給變量,并存在實例變量vars中。
				this.vars[varIndex] = result;
				//返回表達式的值
				return result;
			}
		}
		//如果當前標記類型不是變量,或者不是賦值語句,則用加減法計算表達式的值。
		return this.parseAddOrSub();
	}

	/**
	 * 計算加減法表達式
	 */ 
	private double parseAddOrSub() throws Exception {
		char op;//操作符
		double result;//結果
		double partialResult;//子表達式的結果
		//用乘除法計算當前子表達式的值
		result = this.parseMulOrDiv();
		//如果當前標記的第一個字母是加減號,則繼續進行加減法運算。
		while ((op = this.token.charAt(0)) == '+' || op == '-') {
			//取下一個標記
			this.getToken();
			//用乘除法計算當前子表達式的值
			partialResult = this.parseMulOrDiv();
			switch (op) {
			case '-':
				//如果是減法,則用已處理的子表達式的值減去當前子表達式的值
				result = result - partialResult;
				break;
			case '+':
				//如果是加法,用已處理的子表達式的值加上當前子表達式的值
				result = result + partialResult;
				break;
			}
		}
		return result;
	}

	/**
	 * 計算乘除法表達式,包括取模運算
	 */ 
	private double parseMulOrDiv() throws Exception {
		char op;//操作符
		double result;//結果
		double partialResult;//子表達式的結果
		//用指數運算計算當前子表達式的值
		result = this.parseExponent();
		//如果當前標記的第一個字母是乘、除或者取模運算符,則繼續進行乘除法運算。
		while ((op = this.token.charAt(0)) == '*' || op == '/' || op == '%') {
			//取下一個標記
			this.getToken();
			//用指數運算計算當前子表達式的值
			partialResult = this.parseExponent();
			switch (op) {
			case '*':
				//如果是乘法,則用已處理子表達式的值乘以當前子表達式的值
				result = result * partialResult;
				break;
			case '/':
				//如果是除法,先判斷當前子表達式的值是否為0,如果為0,則拋出被0除異常
				//除數不能為0
				if (partialResult == 0.0){
					this.handleError(DIVBYZERO_ERROR);
				}
				//除數不為0,則進行除法運算
				result = result / partialResult;
				break;
			case '%':
				//如果是取模運算,也要判斷當前子表達式的值是否為0
				//如果為0,則拋出被0除異常
				if (partialResult == 0.0){
					this.handleError(DIVBYZERO_ERROR);
				}
				//進行取模運算
				result = result % partialResult;
				break;
			}
		}
		return result;
	}

	/**
	 * 計算指數表達式
	 * @throws Exception
	 */  
	private double parseExponent() throws Exception {
		double result;//結果
		double partialResult;//子表達式的值
		double ex;//指數的底數
		int t;//指數的冪

		//用一元運算計算當前子表達式的值(底數)
		result = this.parseUnaryOperator();
		//如果當前標記為"^"運算符,則為指數計算
		if (this.token.equals("^")) {
			//獲取下一個標記,即獲取指數的冪
			this.getToken();
			partialResult = this.parseExponent();
			ex = result;
			if (partialResult == 0.0) {
				//如果指數的冪為0,則指數的值為1
				result = 1.0;
			} else {
				//否則,指數的值為個數為指數冪的底數相乘的結果。
				for (t = (int) partialResult - 1; t > 0; t--){
					result = result * ex;
				}
			}
		}
		return result;
	}

	/**
	 * 計算一元運算,+,-,表示正數和復數
	 */  
	private double parseUnaryOperator() throws Exception {
		double result;//結果
		String op;//操作符

		op = "";
		//如果當前標記類型為分隔符,而且分隔符的值等于+或者-。
		if ((this.tokenType == DELIMITER_TOKEN) && 
				this.token.equals("+") || this.token.equals("-")) {
			op = this.token;
			this.getToken();
		}
		//用括號運算計算當前子表達式的值
		result = this.parseBracket();
		if (op.equals("-")){
			//如果操作符為-,則表示負數,將子表達式的值變為負數
			result = -result;
		}

		return result;
	}

	/**
	 * 計算括號運算
	 */  
	private double parseBracket() throws Exception {
		double result;//結果
		//如果當前標記為左括號,則表示是一個括號運算
		if (this.token.equals("(")) {
			//取下一個標記
			this.getToken();
			//用加減法運算計算子表達式的值
			result = this.parseAddOrSub();
			//如果當前標記不等于右括號,拋出括號不匹配異常
			if (!this.token.equals(")")){
				this.handleError(UNBALPARENS_ERROR);
			}
			//否則取下一個標記
			this.getToken();
		} else {
			//如果當前標記不是左括號,表示不是一個括號運算,則用原子元素運算計算子表達式的值
			result = this.parseAtomElement();
		}

		return result;
	}

	/**
	 * 計算原子元素運算,包括變量和數字
	 * @return
	 * @throws Exception
	 */  
	private double parseAtomElement() throws Exception {
		double result = 0.0;//結果
		
		switch (this.tokenType) {
		case NUMBER_TOKEN:
			//如果當前標記類型為數字
			try {
				//將數字的字符串轉換成數字值
				result = Double.parseDouble(this.token);
			} catch (NumberFormatException exc) {
				this.handleError(SYNTAX_ERROR);
			}
			//取下一個標記
			this.getToken();
			break;
		case VARIABLE_TOKEN:
			//如果當前標記類型是變量,則取變量的值
			result = this.findVar(token);
			this.getToken();
			break;
		default:
			this.handleError(SYNTAX_ERROR);
			break;
		}
		return result;
	}

	/**
	 * 根據變量名獲取變量的值,如果變量名長度大于1,則只取變量的第一個字符
	 * @param vname	變量名
	 * @throws Exception
	 */ 
	private double findVar(String vname) throws Exception {
		//如果變量的第一個字符不是字母,則拋出語法異常
		if (!Character.isLetter(vname.charAt(0))) {
			handleError(SYNTAX_ERROR);
			return 0.0;
		}
		//從實例變量數組vars中取出該變量的值
		return vars[Character.toUpperCase(vname.charAt(0)) - 'A'];
	}

	/**
	 * 回滾,將解析器當前指針往前移到當前標記位置
	 */
	private void putBack() {
		if (this.token == EOE){
			return;
		}
		//解析器當前指針往前移動
		for (int i = 0; i < this.token.length(); i++) {
			this.expIndex--;
		}
	}

	/**
	 * 處理異常情況
	 * @param errorType	錯誤類型
	 * @throws Exception
	 */
	private void handleError(int errorType) throws Exception {
		//遇到異常情況時,根據錯誤類型,取得異常提示信息,將提示信息封裝在異常中拋出
		//可以考慮用自定義異常,而不用Exception
		throw new Exception(ERROR_MESSAGES[errorType]);
	}

	/**
	 * 獲取下一個標記
	 */  
	private void getToken() {
		//初始值
		this.tokenType = NONE_TOKEN;
		this.token = "";

		// 檢查表達式是否結束
		// 如果解析器當前指針已經到達了字符串的長度,則表明表達式已經結束,置當前標記的置為EOE
		if (this.expIndex == this.exp.length()) {
			this.token = EOE;
			return;
		}

		// 跳過表達式中的空白符 
		while (this.expIndex < this.exp.length()
				&& Character.isWhitespace(this.exp.charAt(this.expIndex))) {
			++this.expIndex;
		}

		// 再次檢查表達式是否結束 
		if (this.expIndex == this.exp.length()) {
			this.token = EOE;
			return;
		}
		
		//取得解析器當前指針指向的字符
		char currentChar = this.exp.charAt(this.expIndex);
		//如果當前字符是一個分隔符,則認為這是一個分隔符標記,給當前標記和標記類型賦值,并將指針后移
		if (isDelim(currentChar)) {  
			this.token += currentChar;
			this.expIndex++;
			this.tokenType = DELIMITER_TOKEN;
			
		} else if (Character.isLetter(currentChar)) {
			//如果當前字符是一個字母,則認為是一個變量標記。
			//將解析器指針往后移,直到遇到一個分隔符,之間的字符都是變量的組成部分
			while (!isDelim(currentChar)) {
				this.token += currentChar;
				this.expIndex++;
				if (this.expIndex >= this.exp.length()) {
					break;
				} else {
					currentChar = this.exp.charAt(this.expIndex);
				}
			}
			//設置標記類型為變量
			this.tokenType = VARIABLE_TOKEN;
			
		} else if (Character.isDigit(currentChar)) { 
			// 如果當前字符是一個數字,則認為當前標記的類型為數字
			// 將解析器指針往后移,直到遇到一個分隔符,之間的字符都是該數字的組成部分
			while (!isDelim(currentChar)) {
				this.token += currentChar;
				this.expIndex++;
				if (this.expIndex >= this.exp.length()){
					break;
				} else {
					currentChar = this.exp.charAt(this.expIndex);
				}
			}
			//設置標記類型為數字
			this.tokenType = NUMBER_TOKEN;
		} else { 
			//無法識別的字符,則認為表達式結束
			this.token = EOE;
			return;
		}
	}

	/**
	 * 判斷一個字符是否為分隔符。
	 * 表達式中的字符包括: 
	 * 加"+"、減"-"、乘"*"、除"/"、取模"%"、指數"^"、賦值"="、左括號"("、右括號")"
	 * @param c
	 * @return
	 */ 
	private boolean isDelim(char c) {
		if ((" +-/*%^=()".indexOf(c) != -1))
			return true;
		return false;
	}

	public static void main(String[] args) throws Exception {
		ExpressionParser test = new ExpressionParser();
		String exp1 = "a = 5.0";
		System.out.println("exp1(\"a = 4.0\") = " + test.evaluate(exp1));
		String exp2 = "b = 3.0";
		System.out.println("exp2(\"b = 3.0\") = " + test.evaluate(exp2));

		String exp3 = "(a+b) * (a-b)";
		System.out.println("exp3(\"(a+b) * (a-b)\") = " + test.evaluate(exp3));

		String exp4 = "3*5-4/2";
		System.out.println("exp4(\"3*5-4/2\") = " + test.evaluate(exp4));

		String exp5 = "(4-2)*((a+b)/(a-b))";
		System.out.println("exp5(\"(4-2)*((a+b)/(a-b))\") = " + test.evaluate(exp5));
		
		String exp6 = "5 % 2";
		System.out.println("exp6(\"5%2\") = " + test.evaluate(exp6));
		String exp7 = "3^2 * 5 + 4";
		System.out.println("exp7(\"3^2 * 5 + 4\") = " + test.evaluate(exp7));
		
		/**
		 * 一個簡單的表達式根據運算時的優先級從高到底為:
		 * (1)原子元素表達式,包括數字和變量;
		 * (2)括號表達式;
		 * (3)一元表達式,取數的負數;
		 * (4)指數表達式;
		 * (5)乘、除、取模表達式;
		 * (6)加、減表達式
		 * (7)賦值表達式;
		 * 因此,在計算一個表達式的值時,應該按優先級從高到底進行運算。
		 * 在本程序中,每個優先級的表達式的運算都用一個私有方法實現。在私有方法中,首先計算更高優先級的表達式。
		 * 即采用了類似遞歸調用的方式,盡管在evaluate方法中最先調用的是優先級最低的表達式的值,
		 * 但在實質上卻是優先級最高的表達式的私有方法最先被執行完。這就保證了表達式的運算是按照優先級從高到底的順序執行的。
		 */
	}
}

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
欧美成人女星排名| 免费在线看成人av| 日韩国产欧美三级| aaa欧美大片| 欧美va亚洲va在线观看蝴蝶网| 亚洲人成在线播放网站岛国 | 欧美国产精品一区| 香蕉成人啪国产精品视频综合网| 国产曰批免费观看久久久| 在线观看亚洲一区| 国产精品亲子伦对白| 久久精品国产精品亚洲红杏| 欧洲一区在线电影| 国产精品嫩草99a| 国内精品国产成人国产三级粉色 | 国产精品99久久久久久有的能看 | 日本免费新一区视频| 91官网在线观看| 综合激情成人伊人| 国产91对白在线观看九色| 欧美大片一区二区| 免费成人美女在线观看.| 欧美日韩高清影院| 亚洲国产精品欧美一二99| 成人黄色在线视频| 国产精品久久久久久户外露出| 国产一区二区不卡老阿姨| 日韩一区二区三区免费看 | 日韩一区在线免费观看| 国产麻豆精品视频| 久久久久高清精品| 国产精品白丝jk黑袜喷水| 亚洲免费观看高清| 在线观看一区二区视频| 亚洲午夜久久久久久久久久久 | 国产毛片精品一区| 久久免费国产精品| 国产精品资源网| 欧美激情一区二区三区不卡| 成人av午夜电影| 国产精品久久久久久久久图文区 | 国产精品久久久久7777按摩| 成人av电影在线播放| 国产精品色婷婷| 97se亚洲国产综合自在线| 亚洲免费电影在线| 欧美午夜电影在线播放| 午夜电影久久久| 精品国产a毛片| 成人精品一区二区三区四区 | 国产欧美一区二区三区网站| 成人小视频在线观看| 最新久久zyz资源站| 欧美中文一区二区三区| 日本不卡视频一二三区| 亚洲一区在线观看免费 | 欧美精品一区视频| 国产成人免费视频精品含羞草妖精| 久久久精品tv| 一本色道久久综合亚洲aⅴ蜜桃| 亚洲一区二区五区| 精品欧美一区二区在线观看| 懂色av一区二区三区蜜臀| 亚洲精品一二三| 2021国产精品久久精品| 成人综合在线视频| 亚洲v精品v日韩v欧美v专区 | 日韩一区二区在线观看视频| 国产九九视频一区二区三区| 亚洲免费观看视频| 7777精品伊人久久久大香线蕉超级流畅 | 欧美一区二区三区四区五区| 国产成人精品免费一区二区| 亚洲一区在线视频观看| 国产午夜久久久久| 欧美日韩一区三区四区| 成人精品亚洲人成在线| 美女国产一区二区| 自拍偷拍欧美精品| 精品对白一区国产伦| 欧美又粗又大又爽| 国产福利精品一区二区| 视频在线观看一区二区三区| 国产精品免费看片| 日韩欧美的一区二区| 欧美午夜精品理论片a级按摩| 国产剧情一区二区| 日韩福利电影在线| 尤物在线观看一区| 欧美激情资源网| 精品国产精品网麻豆系列| 欧美丝袜丝nylons| 色先锋资源久久综合| 高清不卡一二三区| 国产一区久久久| 日本不卡不码高清免费观看| 亚洲午夜私人影院| 一区二区三区在线看| 最近中文字幕一区二区三区| 久久免费美女视频| 日韩欧美中文字幕公布| 欧美日韩黄色影视| 欧美自拍丝袜亚洲| 色婷婷综合久久久中文一区二区| 国模一区二区三区白浆| 美脚の诱脚舐め脚责91| 日韩电影在线一区二区三区| 亚洲综合色婷婷| 亚洲三级久久久| 中文字幕一区二| 国产精品电影院| 国产精品福利影院| 中文字幕二三区不卡| 日本一区二区免费在线| 久久久午夜精品理论片中文字幕| 精品国产一区二区三区忘忧草 | 欧美无乱码久久久免费午夜一区| 成人禁用看黄a在线| av在线不卡免费看| 99久久婷婷国产综合精品| 成+人+亚洲+综合天堂| 成人av在线资源网站| 91小视频在线免费看| 在线视频国产一区| 欧美高清视频在线高清观看mv色露露十八 | 久久人人97超碰com| 26uuu精品一区二区在线观看| 精品久久久久久久久久久院品网 | 这里只有精品99re| 日韩亚洲欧美综合| 精品处破学生在线二十三| 久久久久国产精品人| 亚洲婷婷在线视频| 偷拍与自拍一区| 精品一区二区三区在线观看 | 久久国产剧场电影| 国产一区二区免费在线| 95精品视频在线| 欧美精品少妇一区二区三区| 欧美一级在线视频| 久久久久99精品一区| 亚洲女同ⅹxx女同tv| 婷婷夜色潮精品综合在线| 国内精品免费**视频| 91在线国产福利| 91精品在线免费观看| 国产色一区二区| 亚洲国产精品一区二区www在线 | 国产成+人+日韩+欧美+亚洲| 色婷婷亚洲婷婷| 日韩午夜激情av| 成人欧美一区二区三区在线播放| 亚洲最新视频在线观看| 国产一区在线观看视频| 色成年激情久久综合| 精品美女被调教视频大全网站| 亚洲欧美视频在线观看视频| 久久精品国产999大香线蕉| 99re成人精品视频| 日韩精品最新网址| 亚洲激情中文1区| 国产一区二区三区| 欧美精品1区2区| 亚洲美女免费在线| 国产精品911| 777亚洲妇女| 亚洲免费视频成人| 国产乱子轮精品视频| 色综合天天综合色综合av| 日韩欧美精品在线| 樱桃国产成人精品视频| 国产一区二区三区精品视频| 欧美日韩精品一区二区三区| 久久久久久久久久美女| 亚洲国产一区二区三区| 成人av网址在线| 精品久久免费看| 日本不卡在线视频| 欧美在线免费观看亚洲| 国产精品另类一区| 国产精品自产自拍| 日韩写真欧美这视频| 亚洲福利国产精品| 成人动漫av在线| 国产三级精品三级| 久久精品99国产精品| 欧美久久久一区| 亚洲成人黄色小说| 欧美日韩三级一区二区| 亚洲男人的天堂网| 91网页版在线| **性色生活片久久毛片| 成人永久看片免费视频天堂| 久久综合色婷婷| 男女男精品视频网| 日韩一区二区三区在线视频| 午夜精品一区二区三区三上悠亚 | 亚洲精品在线电影| 美腿丝袜亚洲一区| 欧美高清精品3d|