電腦思維和人的思維的不同 對於一個算式3+2*(4-3)/5 人的思維是根據括弧和符號優先順序,優先計算括弧中的數據,在進行乘法和除法,在處理加法運算 但是電腦的思維是線性的,電腦會按照算式的前後順序,從前往後進行運算,這樣會導致運算結果錯誤 電腦如何套用人的運算思維 想要讓電腦具有人的”思 ...
電腦思維和人的思維的不同
對於一個算式3+2*(4-3)/5
人的思維是根據括弧和符號優先順序,優先計算括弧中的數據,在進行乘法和除法,在處理加法運算
但是電腦的思維是線性的,電腦會按照算式的前後順序,從前往後進行運算,這樣會導致運算結果錯誤
電腦如何套用人的運算思維
想要讓電腦具有人的”思維“,就需要使用棧,將數據和運算符號之間的順序按照電腦可以理解的方式排列
想要改變規則,就需要將人理解的中綴表達式轉換為尾碼表達式
轉化的規則是:
- 中綴表達式轉化成尾碼表達式
1.遇到操作數直接放入到集合中
2.遇到操作符
2.1當棧為空,或棧頂元素為(,直接放入到棧中
2.2當優先順序比棧頂元素高時,直接進棧
2.3當優先順序小於等於棧頂元素,將棧頂元素出棧放入到集合中,再和棧頂元素比較
3.遇到左右括弧
3.1如果為左括弧,直接加入棧
3.2如果為右括弧,依次將棧頂元素彈出,直到左括弧,並將左括弧彈出
4.最後將棧頂元素依次彈出
中綴表達式轉換為尾碼表達式的過程
以運算算式 3-(4-3)/5*10
以中綴表達式表示為 3 - 4 - 3 / 5 * 10
尾碼表達式表示為 3 4 3 - 5 / 10 * -
其中涉及到了棧的入棧和出棧,
轉化過程:
1.先創建一個集合,用於記錄尾碼表達式,創建一個棧,用於記錄運算符
2.3進入集合,-進入棧 集合 3 棧 -
3.左括弧進棧 集合3 棧- (
4.4進入集合,-進入棧,與此時的棧頂元素比較,此時棧頂元素是左括弧,-直接放入棧中 集合3 4 棧 - ( -
5.3進入集合,-和棧頂元素比較,優先順序相等或者小於,將棧頂元素-彈出 集合中為 3 4 3 - 棧-(
6.右括弧進棧 將棧頂元素彈出,直到左括弧並彈出左括弧 集合3 4 3 - 棧-
7./進棧,優先順序大於-,直接進棧,5 進入集合 集合3 4 3 - 5 棧- /
8.*進棧,優先等於棧頂元素,彈出棧頂元素,10進入集合 集合 3 4 3 - 5 / 10 棧 - *
9.算式獲取完成,將棧中元素全部彈出 集合 3 4 3 - 5 / 10 * -
實現
整數算式運算
package com.prettyspider.calculate;
import java.util.*;
/**
* @author prettyspider
* @ClassName CalcInt
* @description: 傳入整數運算字元串,計算其數值
* @date 2023/9/11 6:21
* @Version V1.0
*/
public class CalcInt {
/**
* 中綴表達式轉化成尾碼表達式
* <p>
* 1.遇到操作數直接放入到集合中
* 2.遇到操作符
* 2.1當棧為空,或棧頂元素為(,直接放入到棧中
* 2.2當優先順序比棧頂元素高時,直接進棧
* 2.3當優先順序小於等於棧頂元素,將棧頂元素出棧放入到集合中,再和棧頂元素比較
* 3.遇到左右括弧
* 3.1如果為左括弧,直接加入棧
* 3.2如果為右括弧,依次將棧頂元素彈出,直到左括弧,並將左括弧彈出
* 4.最後將棧頂元素依次彈出
*/
public static void main(String[] args) {
// 要傳入的運算字元串
String s = "10+2*(3-4)+20*(3*4+3)/3+20";
// 中綴表達式轉化成尾碼表達式
List<String> list = inToPastExpression(s);
// 計算
int number = calc(list);
System.out.println("計算的數值為"+number);
}
/**
* 計算尾碼表達式
* @param list 尾碼表達式集合
* @return 傳入的整數運算字元串的計算數值
*/
private static int calc(List<String> list) {
// 創建棧,用於記錄數據
Stack<String> stack = new Stack<>();
for (String s : list) {
// 1.放入 當是數值時,直接放入棧中
if (s.matches("\\d+")) {
stack.push(s);
} else {
// 2.去除 當是運算符時,再棧中取出兩個數,進行運算,並將運算之後的結果放入到棧中
int num1, num2;
switch (s) {
case "+":
num1 = Integer.parseInt(stack.pop());
num2 = Integer.parseInt(stack.pop());
stack.push(String.valueOf(num2 + num1));
break;
case "-":
num1 = Integer.parseInt(stack.pop());
num2 = Integer.parseInt(stack.pop());
stack.push(String.valueOf(num2 - num1));
break;
case "*":
num1 = Integer.parseInt(stack.pop());
num2 = Integer.parseInt(stack.pop());
stack.push(String.valueOf(num2 * num1));
break;
case "/":
num1 = Integer.parseInt(stack.pop());
num2 = Integer.parseInt(stack.pop());
stack.push(String.valueOf(num2 / num1));
break;
}
}
}
return Integer.parseInt(stack.pop());
}
/**
* 將傳入的字元串進行切分,存放到集合中
* @param str 傳入的算數字元串
* @return 尾碼表達式集合
*/
private static List<String> inToPastExpression(String str) {
// 創建集合和棧
List<String> list = new ArrayList<>();
Stack<String> stack = new Stack<>();
// 切分數據
ArrayList<String> arr = cutString(str);
for (int i = 0; i < arr.size(); i++) {
// * 1.遇到操作數直接放入到集合中
String value = arr.get(i);
if (value.matches("\\d+")) {
list.add(value);
}
// * 3.遇到左右括弧
// * 3.1如果為左括弧,直接加入群
else if ("(".equals(value)) {
stack.push(value);
}
// * 3.2如果為右括弧,依次將棧頂元素彈出,直到左括弧,並將左括弧彈出
else if (")".equals(value)) {
while (!"(".equals(stack.peek())) {
list.add(stack.pop());
}
stack.pop();
}
// * 2.遇到操作符
else {
// * 2.1當棧為空,或棧頂元素為(,直接放入到棧中
// * 2.3當優先順序小於等於棧頂元素,將棧頂元素出棧放入到集合中,再和棧頂元素比較
while (stack.size() != 0 && !judgeOperator(value, stack.peek())) {
list.add(stack.pop());
}
// * 2.2當優先順序比棧頂元素高時,直接進棧
stack.push(value);
}
}
// * 4.最後將棧頂元素依次彈出
while (!stack.empty()) {
list.add(stack.pop());
}
System.out.println(list);
return list;
}
/**
* 將運算字元切分切分成集合
* @param str 待切分字元串
* @return ArrayList<String> 被切分之後的字元串集合
*/
private static ArrayList<String> cutString(String str) {
String[] arr = new String[str.length()-1];
// 為字元串數組賦初值
for (int i = 0; i < arr.length; i++) {
arr[i] = "";
}
int index = 0;
arr[index] = String.valueOf(str.charAt(0));
for (int i = 1; i < str.length(); i++) {
// 1.是數值放入
if (Character.isDigit(str.charAt(i))) {
// 1.1並看其起一個數據是不是也數值,是數值,將其前一個數據*10+本身
if (arr[index].matches("\\d+")) {
arr[index] = String.valueOf(Integer.parseInt(arr[index]) * 10 + Integer.parseInt(String.valueOf(str.charAt(i))));
} else {
arr[++index] = String.valueOf(str.charAt(i));
}
} else {
// 2.不是數值,直接加入到集合中
arr[++index] = String.valueOf(str.charAt(i));
}
}
// 去除null
ArrayList<String> list = removeNull(arr);
return list;
}
/**
* 將字元串數組中為空的元素去除
* @param arr 字元串數組
* @return 去除控制的字元串集合
*/
private static ArrayList<String> removeNull(String[] arr) {
ArrayList<String> list = new ArrayList<>();
for (String s : arr) {
if (!"".equals(s)) {
list.add(s);
}
}
return list;
}
/**
* 比較新舊操作符的權重,判斷是存放還是取出
* @param s1 新的操作符
* @param s2 舊的操作符
* @return 新舊操作符的權重比較
*/
private static boolean judgeOperator(String s1, String s2) {
int num1 = 0, num2 = 0;
switch (s1) {
case "+":
case "-":
num1 = 1;
break;
case "*":
case "/":
num1 = 2;
break;
}
switch (s2) {
case "+":
case "-":
num2 = 1;
break;
case "*":
case "/":
num2 = 2;
break;
}
// 判斷新舊操作符的優先順序
if (num1 > num2) {
return true;
}
return false;
}
}
小數算式運算
小數運算要考慮小數點,相對於整數要比較的複雜
在獲取嫻熟時,需要對小數點的位置進行判斷,對小數點右邊的數據進行處理
package com.prettyspider.calculate;
import java.util.*;
/**
* @author prettyspider
* @ClassName CalcInt
* @description: 傳入整數運算字元串,計算其數值
* @date 2023/9/11 6:21
* @Version V1.0
*/
public class CalcDouble {
/**
* 中綴表達式轉化成尾碼表達式
* <p>
* 1.遇到操作數直接放入到集合中
* 2.遇到操作符
* 2.1當棧為空,或棧頂元素為(,直接放入到棧中
* 2.2當優先順序比棧頂元素高時,直接進棧
* 2.3當優先順序小於等於棧頂元素,將棧頂元素出棧放入到集合中,再和棧頂元素比較
* 3.遇到左右括弧
* 3.1如果為左括弧,直接加入棧
* 3.2如果為右括弧,依次將棧頂元素彈出,直到左括弧,並將左括弧彈出
* 4.最後將棧頂元素依次彈出
*/
public static void main(String[] args) {
// 要傳入的運算字元串
String s = "10.32*3.23+2.234";
// 中綴表達式轉化成尾碼表達式
List<String> list = inToPastExpression(s);
// 計算
double number = calc(list);
System.out.println("計算的數值為"+number);
}
/**
* 計算尾碼表達式
* @param list 尾碼表達式集合
* @return 傳入的整數運算字元串的計算數值
*/
private static double calc(List<String> list) {
// 創建棧,用於記錄數據
Stack<String> stack = new Stack<>();
for (String s : list) {
// 1.放入 當是數值時,直接放入棧中
if (s.matches("(\\d|\\.)+")) {
stack.push(s);
} else {
// 2.去除 當是運算符時,再棧中取出兩個數,進行運算,並將運算之後的結果放入到棧中
double num1, num2;
switch (s) {
case "+":
num1 = Double.valueOf(stack.pop());
num2 = Double.valueOf(stack.pop());
stack.push(String.valueOf(num2 + num1));
break;
case "-":
num1 = Double.valueOf(stack.pop());
num2 = Double.valueOf(stack.pop());
stack.push(String.valueOf(num2 - num1));
break;
case "*":
num1 = Double.valueOf(stack.pop());
num2 = Double.valueOf(stack.pop());
stack.push(String.valueOf(num2 * num1));
break;
case "/":
num1 = Double.valueOf(stack.pop());
num2 = Double.valueOf(stack.pop());
stack.push(String.valueOf(num2 / num1));
break;
}
}
}
return Double.valueOf(stack.pop());
}
/**
* 將傳入的字元串進行切分,存放到集合中
* @param str 傳入的算數字元串
* @return 尾碼表達式集合
*/
private static List<String> inToPastExpression(String str) {
// 創建集合和棧
List<String> list = new ArrayList<>();
Stack<String> stack = new Stack<>();
// 切分數據
ArrayList<String> arr = cutString(str);
for (int i = 0; i < arr.size(); i++) {
// * 1.遇到操作數直接放入到集合中
String value = arr.get(i);
if (value.matches("(\\d|\\.)+")) {
list.add(value);
}
// * 3.遇到左右括弧
// * 3.1如果為左括弧,直接加入群
else if ("(".equals(value)) {
stack.push(value);
}
// * 3.2如果為右括弧,依次將棧頂元素彈出,直到左括弧,並將左括弧彈出
else if (")".equals(value)) {
while (!"(".equals(stack.peek())) {
list.add(stack.pop());
}
stack.pop();
}
// * 2.遇到操作符
else {
// * 2.1當棧為空,或棧頂元素為(,直接放入到棧中
// * 2.3當優先順序小於等於棧頂元素,將棧頂元素出棧放入到集合中,再和棧頂元素比較
while (stack.size() != 0 && !judgeOperator(value, stack.peek())) {
list.add(stack.pop());
}
// * 2.2當優先順序比棧頂元素高時,直接進棧
stack.push(value);
}
}
// * 4.最後將棧頂元素依次彈出
while (!stack.empty()) {
list.add(stack.pop());
}
return list;
}
/**
* 將運算字元切分切分成集合
* @param str 待切分字元串
* @return ArrayList<String> 被切分之後的字元串集合
*/
private static ArrayList<String> cutString(String str) {
String[] arr = new String[str.length()-1];
// 為字元串數組賦初值
for (int i = 0; i < arr.length; i++) {
arr[i] = "";
}
int index = 0;
arr[index] = String.valueOf(str.charAt(0));
for (int i = 1; i < str.length(); i++) {
// 1.是數值放入
if (Character.isDigit(str.charAt(i))|| ".".equals(String.valueOf(str.charAt(i)))) {
// 1.1並看其起一個數據是不是也數值,是數值,將其前一個數據拼接
if (arr[index].matches("(\\d|\\.)+")) {
arr[index] = arr[index] + str.charAt(i);
} else {
arr[++index] = String.valueOf(str.charAt(i));
}
} else {
// 2.不是數值,直接加入到集合中
arr[++index] = String.valueOf(str.charAt(i));
}
}
// 去除null
ArrayList<String> list = removeNull(arr);
return list;
}
/**
* 將字元串數組中為空的元素去除
* @param arr 字元串數組
* @return 去除控制的字元串集合
*/
private static ArrayList<String> removeNull(String[] arr) {
ArrayList<String> list = new ArrayList<>();
for (String s : arr) {
if (!"".equals(s)) {
list.add(s);
}
}
return list;
}
/**
* 比較新舊操作符的權重,判斷是存放還是取出
* @param s1 新的操作符
* @param s2 舊的操作符
* @return 新舊操作符的權重比較
*/
private static boolean judgeOperator(String s1, String s2) {
int num1 = 0, num2 = 0;
switch (s1) {
case "+":
case "-":
num1 = 1;
break;
case "*":
case "/":
num1 = 2;
break;
}
switch (s2) {
case "+":
case "-":
num2 = 1;
break;
case "*":
case "/":
num2 = 2;
break;
}
// 判斷新舊操作符的優先順序
if (num1 > num2) {
return true;
}
return false;
}
}
在獲取數據時,巧妙的使用正則獲取對應的數據,利於數據的處理