重温:数据结构与算法 - 06栈

xwzz.jpg

重温:数据结构与算法 - 开篇
重温:数据结构与算法 - 复杂度分析(一)
重温:数据结构与算法 - 复杂度分析(二)
重温:数据结构与算法 - 数组
重温:数据结构与算法 - 链表(一)
重温:数据结构与算法 - 链表(二)

叨叨两句

说好的尽量日更o(╥﹏╥)o~,太懒了,最后两个练习题一直没敲,拖了两天才更,啪 啪~ 打脸!

什么是栈

本章介绍一种“操作受限”的数据结构,它仅支持:

  • 添加操作(入栈 - push)
  • 删除操作(出栈 - pop)

并具备“先进后出,后进先出”特点:

子弹匣.png

图中,正在装填的弹匣就是典型的栈结构,先装入的子弹被压入了底部,后装入的在顶部,当使用时优先使用顶部子弹。那么,在内存中给予这种模式存储数据的容器,称之为:

内存结构图.png

栈的实现

根据定义,很容抽象出一个只能新增、删除的操作接口来:

public interface Stack<T> {
  //压栈
  boolean push(T item);

  //弹栈
  T pop();
  
  // 返回栈顶数据,不弹栈
  T peek();

  //是否为空
  boolean isEmpty();

  //栈中数据的数量
  int size();

  //清空
  void clear();
}

有了操作标准,拿什么来存储实际数据?

  • 数组 ?
  • 链表 ?

细想一下,无论是使用数组还是链表存储数据,只要出栈、入栈规则满足“先进后出、后进先出”,那它就是我们想要的栈型结构,当然不同数据结构实现的栈,底层肯定还是具备其本身的数据结构特性的,例如数组实现的栈型结构内存是连续的,链表则是不连续的,我们把数组实现的栈,称之:顺序栈 ; 链表实现则为:链式栈

下面以数组为例,实现一个栈容器:

public class ArrayStack<T> implements Stack<T> {

   private static final int DEFAULT_SIZE = 10;

  // 栈中数据数量
  private int count;

  // 栈中数据(数组实现)
  private Object[] items;
  private Object curItem;

  // 当前栈能存储的最大容量
  private int maxSize;

  public ArrayStack() {
    this(DEFAULT_SIZE);
  }

  public ArrayStack(int initSize) {
    this.items = new Object[initSize];
    this.maxSize = initSize;
  }

  @Override
  public boolean push(T item) {
    // 入栈
    if (count == maxSize) {
        //扩容
        grow();
    }
    items[count] = item;
    ++count;
    curItem = item;
    return true;
  }

  private void grow() {
    Object[] temp = items;
    items = new Object[temp.length * 2];
    for (int i = 0; i < temp.length; i++) {
        items[i] = temp[i];
    }
    maxSize = items.length;
  }

  @Override
  public T pop() {
    // 出栈
    if (count == 0) return null;
    T item = (T) items[count - 1];
    items[count - 1] = null;
    --count;
    if (count == 0) {
        curItem = null;
    } else {
        curItem = items[count - 1];
    }
    return item;
  }

  @Override
  public T peek() {
    return (T) curItem;
  }

  @Override
  public boolean isEmpty() {
    return count == 0;
  }

  @Override
  public int size() {
    return count;
  }

  @Override
  public void clear() {
    if (isEmpty()) return;
    for (int i = 0; i < size(); i++) {
        items[i] = null;
    }
    count = 0;
  }

  public void print() {
     for (int i = 0; i < count; i++) {
        System.out.print(items[i] + "\t");
    }
    System.out.println();
  }
}

测试:

    ArrayStack<String> stack = new ArrayStack<>();
    stack.push("1");
    stack.push("2");
    stack.push("3");
    stack.push("4");
    stack.push("5");
    stack.print();

    String item1 = stack.pop();
    System.out.println(item1);

    String item2 = stack.pop();
    System.out.println(item2);

    stack.print();

输出:

1   2   3   4   5   
5
4
1   2   3
  • 时间复杂度:

    • 最好时间复杂度:空间够用,直接入栈,复杂度 O(1)

    • 最坏时间复杂度:空间不足,动态扩容,复杂度 O(n)

    • 平均时间复杂度:还记得复杂度分析(二)中提到的平摊分析法吗?随着数据的改变,算法出现某种固定规律,这里明显当执行完n次O(1)插入操作后,都需要进行一次O(n)的数据位移,正好可以把这n次搬运平摊给前面各自1次的插入操作,得到平摊时间复杂度:O(1)

  • 空间复杂度:

    注意,这里存储数据需要一个大小为 n 的数组,并不是说空间复杂度就是 O(n)。因为,这 n 个空间是必须的,无法省掉。我们说空间复杂度的时候,是指除了原本的数据存储空间外,算法运行还需要额外的存储空间。所以栈的插入操作空间复杂度是:O(1)

栈 - 练习1:运算表达式

要求给出一个运算表达式字符串,计算出表达式结果。
输入字符串: 3 + 2 * 5 - 4 / 2
输出 :11

思路:

  • 1、利用两个栈,一个用来保存数字,另一个用来保存运算符,从左向右遍历表达式;
  • 2、当遇到数字,我们就直接压入数字栈;
  • 3、当遇到运算符,就与运算符栈的栈顶元素进行比较,若比运算符栈顶元素优先级高,就将当前运算符压入栈;
  • 4、若比运算符栈顶元素的优先级低或者相同,从运算符栈中取出栈顶运算符,从操作数栈顶取出2个操作数,然后进行计算,把计算完的结果压入数字栈;
  • 5、重复第3步的操作,继续比较栈顶操作符。
private static void computeExp(String exp) {
    ArrayStack<String> numStack = new ArrayStack<>();
    ArrayStack<String> opStack = new ArrayStack<>();
    char[] chars = exp.toCharArray();
    String preItem = null;
    for (int i = 0; i < chars.length; i++) {
        String item = String.valueOf(chars[i]);

        if (isOp(item)) {
            // 操作符
            while (true) {  
                if (opStack.isEmpty()) {
                    // 直接入栈
                    opStack.push(item);
                    break;
                }
                // 比较优先级
                String preOp = opStack.pop();
                if (opLv(item) > opLv(preOp)) {
                    // 大于
                    opStack.push(preOp);
                    opStack.push(item);
                    break;
                } else {
                    // 小于等于 , 取出2个数,和栈顶操作符进行运算
                    String num1 = numStack.pop();
                    String num2 = numStack.pop();
                    int result = compute(num2, num1, preOp);
                    numStack.push(String.valueOf(result));
                }
            }
        } else {
            // 数字
            if (!numStack.isEmpty() && !isOp(preItem)) {
                // 前一个也是数字,拼接后入数字栈
                preItem = numStack.pop();
                item = preItem + item;
            }
            // 入数字栈
            numStack.push(item);
        }
        preItem = item;
    }

    // 清空栈,计算最终结果
    int len =  opStack.size();
    for(int i = 0 ; i < len ; i++){
       String num1 =   numStack.pop();
       String num2 = numStack.pop();
       String op = opStack.pop();
       int result = compute(num2 , num1 , op);
       numStack.push(String.valueOf(result));
    }
    System.out.println( exp + " = " +  numStack.pop());
}

private static int compute(String num2, String num1, String op) {
    int n2 = Integer.parseInt(num2);
    int n1 = Integer.parseInt(num1);
    switch (op) {
        case "+":
            return n2 + n1;
        case "-":
            return n2 - n1;
        case "*":
            return n2 * n1;
        case "/":
            return n2 / n1;
        default:
            throw new RuntimeException("no op");
    }

}

private static boolean isOp(String item) {
    if (item == null) return false;
    return item.matches("[+\\-*/]");
}

private static int opLv(String op) {
    switch (op) {
        case "+":
        case "-":
            return 1;
        case "*":
        case "/":
            return 2;
        default:
            throw new RuntimeException("no op");
    }
}

测试:

    computeExp("3+10*5-8/2");
    computeExp("4+8*5/4-7");
    computeExp("50/2-3*7+1");

输出:

  3+10*5-8/2 = 49
  4+8*5/4-7 = 7
  50/2-3*7+1 = 5

时间复杂度:O(n²)
空间复杂度:O(1)

栈 - 练习2:模拟浏览器前进后退功能

思路:
利用两个栈,X栈用于存储新打开的页面,Y栈存储后退页面
当点击后退时,将X出栈一个页面,并存入Y栈
当点击前进时,将Y出栈一个页面,并存入X栈
当打开新页面时,入X栈,清空Y栈

 private static void go(ArrayStack<String> goStack, ArrayStack<String> backStack) {
    // 当点击前进时,将Y出栈一个页面,并存入X栈
    if(backStack.isEmpty()) {
        System.out.println("go:" + goStack.peek());
        return;
    }
    String page = backStack.pop();
    goStack.push(page);
    System.out.println("go:" + goStack.peek());
}

private static void back(ArrayStack<String> goStack, ArrayStack<String> backStack) {
    // 当点击后退时,将X出栈一个页面,并存入Y栈
    String page = goStack.pop();
    backStack.push(page);
    System.out.println("back:" + goStack.peek());
}

private static void open(String page, ArrayStack<String> goStack, ArrayStack<String> backStack) {
    // 当打开新页面时,入X栈,清空Y栈
    goStack.push(page);
    backStack.clear();
    System.out.println("open:" + goStack.peek());
}

测试:

    ArrayStack<String> goStack = new ArrayStack<>();
    ArrayStack<String> backStack = new ArrayStack<>();
    open("页面1", goStack, backStack);
    open("页面2", goStack, backStack);
    open("页面3", goStack, backStack);
    open("页面4", goStack, backStack);
    open("页面5", goStack, backStack);
    back(goStack, backStack);
    back(goStack , backStack);
    go(goStack, backStack);
    go(goStack, backStack);
    go(goStack, backStack);

输出:

open:页面1
open:页面2
open:页面3
open:页面4
open:页面5
back:页面4
back:页面3
go:页面4
go:页面5
go:页面5

时间复杂度:O(1)
空间复杂度:O(1)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容