数据结构之栈与队列

前言

数组与链表结构能够解决很多一对一的线性表问题,但是在许多情况下,使用最原始的数据结构过于复杂,而且不够安全。数组和链表都可以在任意位置增加和删除数据,但在一些场景下这个机制并不适用。比如我们要做一个排队系统,如果人们可以随意插队,那么所有人都会不满吧?栈与队列,就是在一些场景下对数组或链表的包装,使其严格符合需求,大大增加了安全性。

栈,定义为限定仅在表尾进行插入和删除操作的线性表,也就是不允许插队,仅允许在表尾操作。这就好比我们洗碗后把碗叠在一起时,总是后来者居上,而且取出时也是先取出后来者。这种结构在编程中有很多应用场景,比如编辑文档时的撤销操作,肯定是先撤销最后一步操作,再比如递归函数的调用,也必须保证函数执行的顺序,等等。栈的这种后入先出的特性,称为LIFO(last in first out)

栈一般有大小限制(不可能把碗叠的无限高),当然也不是绝对的。我们以建立一个有大小限制的栈为例,说明栈的使用,和它在增、删、查方面的表现。

1. 顺序结构实现

如果栈的大小限定,我们就可以使用数组来实现它了。

  1. 声明一个空栈
class LinearStack{
    Integer[] stack = new Integer[10];
    // 栈顶,也就是可以操作的队尾
    int top = -1;
}
  1. 增加一条数据

向栈中增加数据也叫做入栈,代码如下:

public void push(Integer data){
    // 处理好边界问题
    if(top<stack.length-1){
        stack[++top] = data;
    }
}
  1. 删除一条数据

从栈中删除数据也叫出栈,代码如下:

public Integer pop(){
    if(top<0) return null;
    
    Integer ret = stack[top];
    stack[top] = null;
    top--;
    return ret;
}
  1. 总结

可以看到,使用数组实现栈时,增删操作时间复杂度都是O(1)

2. 链式结构实现

  1. 声明一个空栈

与单链表不同,栈需要一个指针始终指向最后数据的位置,且它的指针域指向前一个结点。我们有两种思路来实现。

  • 思路一:头插法

在链表头部插入数据,并使top指针指向最新数据,next指向原数据。

  • 思路二:尾插法

在链表尾部插入数据,并使top指针指向最新数据,prev指向原数据。

两种做法实际上是对称的,我们以方法一为例,代码如下:

class LinkStack{
    Node top = new Node(null);
    // 定义长度
    static final int SIZE = 10;
    int len = -1;
}
  1. 增加一条数据
public void push(Integer data){
    if(len<SIZE-1){
        if(top.data==null){
            top.data = data;
        }else{
            Node newNode = new Node(data);
            newNode.next = top;
            top = newNode;
        }
        len++;
    }
}
  1. 删除一条数据
public Integer pop(){
    if(len<0) return null;
    Integer ret = top.data;
    Node next = top.next;
    top.next = null;
    top = next;
    len--;
    return ret;
}
  1. 总结

用链表实现栈时,增删复杂度也都为O(1),但它的优点在于可以动态调整大小,而不会像数组那样需要处理扩容性能问题。栈Stack在JDK中也有实现,大家可以阅读相关源码深入了解。

队列

除了栈这种场景,还有一种场景不能直接使用链表,这就是队列。它定义为只允许在一端进行插入操作,而在另一端进行删除操作的线性表。比如我们买车票要排队,坐公交要排队,有时吃饭人多时还是要排队。在编程中也有很多场景要用到队列,典型的应用场景就是阻塞队列,在多线程编程环境下使用非常频繁。

队列也可以使用数组和链表两种方式来实现。但是由于队列的增加和删除不在一端,这就意味着使用数组实现时,删除操作要移动所有数据,比如有数组如下:

数组

现在要进行出队操作,也就是把数据1删除,后续的5、2、11都要向前移动一个位置,如下所示:

出队

这样一来,出队的时间复杂度就是O(n)。那么,有没有办法解决此问题呢,还是数组不适合用来实现队列?

实际上,这个问题是有解的,那就是使用循环队列。假设有数组如下:

数组

我们向其中插入一些数据,使它全部填充,如下:

填充

现在进行出队操作,也就是删除数据5,与之前不同的是,这次数据不再向前移动,而仅仅把5原先的位置空出来,如下:

5出队

现在队列不是满的了,至少位置0是空闲的。那么我们要入队数据10,应该怎么操作呢?我们把数组看做是循环的,假设位置7的下一个位置是0,也就是把首尾连接起来,如下:

循环数组

这样数据10就可以插入到位置0了,如下:

10入队

要这样使用数组,我们就需要两个标记,一个用来标记队列开始的位置,一个用来标记队列结束的位置,如上图队列的队首在位置1,队尾在位置0,标记如下:

标记

有了标记,对队列的操作就变得容易多了,队列空和满也都可以根据标记来判断。关于循环队列的Java实现,大家可以参考我分析的ArrayDeque源码:Java集合源码分析之Queue(三):ArrayDeque

接下来,我们用单链表来实现一个队列。

  1. 声明一个空队列
class LinkQueue{
    Node front = new Node(null);
    Node rear = new Node(null);
    int len = 0;
}
  1. 增加一条数据
public void enqueue(Integer data){
    // 空表时数据放入front中
    if(front.data == null){
        front.data = data;
        len++;
        return;
    }

    // 第二条数据放在rear中,其他数据向后追加
    if(rear.data==null){
        rear.data= data;
        front.next = rear;
    }else{
        Node newNode = new Node(data);
        rear.next = newNode;
        rear = newNode;
    }
    len++;
}
  1. 删除一条数据
public Integer dequeue(){
    if (front.data == null) {
        return null;
    }
    Integer ret = front.data;
    front = front.next;
    len--;
    return ret;
}

总结

栈与队列都是在数组或链表的基础上改造而来,它们的应用十分广泛,在实际开发使用中,我们应该灵活选用。

以上涉及代码均已上传至我的github


我是飞机酱,如果您喜欢我的文章,可以关注我~

编程之路,道阻且长。唯,路漫漫其修远兮,吾将上下而求索。

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

推荐阅读更多精彩内容