美文网首页
实现一个自己的栈(MyStack),表达式括号匹配检查及运算

实现一个自己的栈(MyStack),表达式括号匹配检查及运算

作者: _Raye | 来源:发表于2017-04-17 19:22 被阅读0次
  • 自己的栈MyStack<E>,使用泛型参数:

import java.util.ArrayList;
import java.util.List;

public class MyStack<E> {
    private List<E> elements = new ArrayList<>();
    
    /**
     * 添加元素(入栈)
     * @param e
     */
    public void push(E e) {
        elements.add(e);
    }
    
    /**
     * 删除栈顶元素(出栈)
     * @return
     */
    public E pop() {
        return elements.remove(elements.size() - 1);
    }
    
    /**
     * 查看栈顶元素
     * @return
     */
    public E peek() {
        return elements.get(elements.size() - 1);
    }
    
    /**
     * 判断栈是否为空
     * @return
     */
    public boolean isEmpty() {
        return elements.size() == 0;
    }
}
  • 表达式括号匹配检查
  • 题目:检查表达式中的括号是否匹配(表达式中可能有圆括号,方括号,花括号)
  • 要点:使用栈的FILO(先进后出)特性
public class Test {

    public static boolean CheckBrackets(String expr) {
        MyStack<Character> stack = new MyStack<>();
        for (int i = 0; i < expr.length(); i++) {
            char temp = expr.charAt(i);
            if (temp == '(' || temp == '[' || temp == '{') {
                stack.push(temp); //遇到左括号,入栈
            }
            else if(temp == ')' || temp == ']' || temp == '}') {
                if (!stack.isEmpty()) { //先判断栈是否为空,如果不为空,则继续进行
                    char left = stack.pop(); //遇到右括号,出栈
                    if ((temp == ')' && left != '(') || 
                            (temp == ']' && left != '[') || 
                            (temp == '}' && left != '{')) {
                        return false; //左括号与右括号不匹配
                    } 
                }
                else {
                    return false; //遇到右括号,但没有左括号与之匹配
                } 
            }
        }
        return stack.isEmpty(); //检查所有左括号是否完成匹配
    }
    
    public static void main(String[] args) {
        System.out.println(CheckBrackets("([3 + 5] * 2) + (1 * 2)"));
    }
}
  • 表达式运算
  • 1.需要先将表达式转为逆波兰表达式
  • 2.通过转为的逆波兰表达式进行运算
package com.qfedu.day01;

class Q002 {

    // 逆波兰表达式 - 中缀表达式转逆波兰表达式
    public static double eval(String suffix) { //参数为逆波兰表达式
        MyStack<Double> stack = new MyStack<>();
        StringBuilder tempStr = new StringBuilder();
        for (int i = 0, len = suffix.length(); i < len; ++i) {
            char temp = suffix.charAt(i);
            if (temp == ' ') {
                if (tempStr.length() > 0) {
                    double num = Double.parseDouble(tempStr.toString());
                    stack.push(num);
                    tempStr.delete(0, tempStr.length());
                }
            }
            else if (temp >= '0' && temp <= '9') {
                tempStr.append(temp);
            }
            else {
                if (stack.size() >= 2) {
                    double num2 = stack.pop();
                    double num1 = stack.pop();
                    double result = 0;
                    switch (temp) {
                    case '+':
                        result = num1 + num2;
                        break;
                    case '-':
                        result = num1 - num2;
                        break;
                    case '*':
                        result = num1 * num2;
                        break;
                    case '/':
                        result = num1 / num2;
                        break;
                    }
                    stack.push(result);
                }
            }
        }
        return stack.pop();
    }

    public static String toSuffix(String expr) { //转为逆波兰表达式
        MyStack<Character> stack = new MyStack<>();
        StringBuilder suffix = new StringBuilder();
        StringBuilder tempStr = new StringBuilder();
        for (int i = 0, len = expr.length(); i < len; ++i) {
            char temp = expr.charAt(i);
            if (temp != ' ') {
                if (temp >= '0' && temp <= '9') {
                    tempStr.append(temp);
                }
                else {
                    if (tempStr.length() > 0) {
                        suffix.append(tempStr.toString() + " ");
                    }
                    tempStr.delete(0, tempStr.length());
                    if (temp == ')' || temp == ']' || temp == '}') {
                        Character op = stack.pop();
                        while (op != '(' && op != '[' && op != '{') {
                            suffix.append(op + " ");
                            if (!stack.isEmpty()) {
                                op = stack.pop();
                            }
                            else {
                                break;
                            }
                        }
                    }
                    else {
                        if (temp != '(' && temp != '[' && temp != '{') {
                            if (!stack.isEmpty()) {
                                Character op = stack.peek();
                                while (!hasLowerOrder(op, temp)) {
                                    suffix.append(stack.pop() + " ");
                                    if (!stack.isEmpty()) {
                                        op = stack.peek();
                                    }
                                    else {
                                        break;
                                    }
                                }
                            }
                        }
                        stack.push(temp);
                    }
                }
            }
        }
        if (tempStr.length() > 0) {
            suffix.append(tempStr.toString() + " ");
        }
        while (!stack.isEmpty()) {
            suffix.append(stack.pop() + " ");
        }
        return suffix.toString();
    }

    private static boolean hasLowerOrder(char op1, char op2) {
        return (op1 == '(' || op1 == '[' || op1 == '{') || ((op1 == '+' || op1 == '-') && (op2 == '*' || op2 == '/'));
    }

    // 栈(stack) - 线性结构 - FILO
    // 队列(queue) - 线性结构 - FIFO
    // 堆(heap) - 层次结构 - 完全二叉树
    public static void main(String[] args) throws Exception {
        System.out.println(eval(toSuffix("2 + 3 * 5")));
        System.out.println(eval(toSuffix("(1 + 2 + 3) * 5")));
        System.out.println(eval(toSuffix("(1 + 2) * 5 * (2 + 3)")));
        System.out.println(eval(toSuffix("(1 + 2) * (5 - 2) * 3)")));
        System.out.println(eval(toSuffix("(1 + 2) / (5 * (2 + 3))")));
        System.out.println(eval(toSuffix("{[1 + 12 + 234]} * 3456 * (45678 - 567890)")));
    }
}

相关文章

  • 实现一个自己的栈(MyStack),表达式括号匹配检查及运算

    自己的栈MyStack,使用泛型参数: 表达式括号匹配检查 题目:检查表达式中的括号是否匹配(表达式中可能有...

  • 栈和队列

    一.栈 栈的作用之一:利用栈后进先出的特点匹配括号,计算带运算符的算法(也就是中缀表达式) 可以把中缀表达式转化为...

  • chap3-栈和队列

    括号匹配问题 // 括号匹配,遇到 '\0' 结束// 遇到花、中、圆左括号进栈,遇到花、中、圆右括号检查栈顶元素...

  • 数据结构与算法 --- 栈习题(Swift)

    首先实现一个顺序存储的栈。 一。括号匹配检验 假设表达式中允许包含两种括号:圆括号和方括号,其嵌套顺序随意,即()...

  • JavaScript数据结构与算法-栈练习

    栈的实现 练习 一. 栈可以用来判断一个算术表达式中的括号是否匹配。编写一个函数,该函数接受一个算术表达式作为参数...

  • 机试常用算法和题型-栈和队列专题

    堆栈+ordermap使用括号匹配 堆栈使用简单计算器 栈+队列实现中缀转后缀,计算后缀表达式 栈+队列计算,包括...

  • 栈: 顺序栈 栈的应用:函数调用栈,表达式求值,括号匹配,浏览器的前进后退。相关code:https://gith...

  • python实现栈和队列及其基本操作

    python实现栈 class MyStack:# 模拟栈 def __init__(self):self.it...

  • 20. 有效的括号

    自己解法 括号匹配问题,第一想法就是使用栈,左括号入栈,遇到匹配的右括号出栈,如果中间有不符合的右括号,直接返回f...

  • 《数据结构与算法Javascript描述》第四章 栈 习题答案

    栈可以用来判断一个算术表达式中的括号是否匹配。编写一个函数,该函数接受一个算术表达式作为参数,返回括号缺失的位置。...

网友评论

      本文标题:实现一个自己的栈(MyStack),表达式括号匹配检查及运算

      本文链接:https://www.haomeiwen.com/subject/eefxzttx.html