首页 > 其他分享 >线性结构中的栈、队列和串是怎么回事?

线性结构中的栈、队列和串是怎么回事?

时间:2023-06-20 11:34:50浏览次数:43  
标签:出栈 怎么回事 队列 元素 栈顶 线性 操作 表达式

一. 栈

1. 栈的概念

(stack) 是一种操作受限的线性表,栈的操作被限定在线性表的尾部进行,栈结构有两个特殊概念:

  • 栈顶:栈的尾部被称为栈顶(Top);
  • 栈底:另一端固定不动,被称为栈底(Bottom)。

栈中的元素只能先入后出最早进入栈的元素所在的位置是栈底,最后进入栈的元素所在的位置是栈顶。数据进入栈的过程叫入栈或压栈,数据从栈中出去的过程叫出栈或弹栈。 如下图所示:
在这里插入图片描述

2. 栈的实现

在实现上,栈既可以用数组来实现,也可以用链表来实现。用数组实现的栈叫顺序栈或静态栈,用链表实现的栈叫做链式栈或动态栈,这两种实现方式分别如下图所示:

3. 栈的操作

刚刚给大家介绍了栈顶和栈底的概念,栈的操作就和这两个概念有关。栈一般只有两个操作:入栈和出栈。

3.1 入栈

入栈操作就是把元素(数据)放入到栈中,入栈操作又称压栈,有时也称之为push操作,均表示入栈操作含义。需要注意的是,根据栈结构的特点,入栈只允许从栈顶放入元素,进入栈的新元素会在最上方,成为栈顶

对于入栈操作的时间复杂度,因为只涉及到一个元素的操作,因此非常简单为O(1)。

3.2 出栈

出栈操作就是把元素(数据)从栈中取出,出栈操作又称弹栈,有时也称之为pop操作,均表示出栈操作含义。同样需要注意,根据栈结构的特点,出栈只需要从栈顶取出元素,元素出栈后,原次顶元素会成为新的栈顶元素

出栈操作的时间复杂度与入栈操作的时间复杂度相同,均是操作栈顶一个元素,因此出栈操作时间复杂度为O(1)。

4. 栈的特点

如上图所示,清晰的展示了栈的入栈操作和出栈操作。可以看到,无论入栈操作,还是出栈操作,都是操作的栈顶元素。

由此,给大家总结栈结构的特点:后进先出,即后进入的元素会先出栈。在计算机术语中,后进先出描述为Last In First Out,简称LIFO;另外也有人表述为先进后出(Frist In Last Out,简称FILO) ,这两者含义其实是相同的。

5. 栈的应用

在实际的应用实践中,我们可以利用栈结构的特殊性和其特点,解决某些特定的问题,此处给大家介绍常见的几个

5.1 子程序调用

程序在执行过程中,不免会涉及到调用。利用栈结构,可以在跳往子程序前,先将下个指令的地址存到堆栈中,直到子程序执行完后再将地址取出,以便回到原来的程序中。

5.2 浏览器前进和后退

我们使用两个栈X和Y,把首次浏览的页面依次压入栈X,当单击后退按钮时,再依次从栈X中出栈,并将出栈的数据依次放入栈Y。当单击前进按钮时,我们依次从栈Y中取出数据, 放入栈X中。当栈X中没有数据时,说明没有页面可以继续后退浏览了。当栈Y中没有数据,那就说明没有页面可以单击前进按钮浏览了。

5.3 表达式求值

首先大家要知道什么是表达式,以及有哪些表达式,才能进一步学习如何进行表达式求值。根据百科的定义:表达式是由数字、算符、数字分组符号(括号)、自由变量和约束变量等以能求得数值的有意义排列方法所得的组合。

听起来非常的抽象和不好理解,我们可以简单地说:由一个或多个操作数通过操作符组合而成的式子就是表达式

比如:3+5、a-b、c==d等这些都是表达式,在这三个表达式中,均包含两个操作数和一个操作符。编程中常见和使用的表达式有:算术表达式、逻辑表达式、字符串表达式等。

表达式按照操作数和操作符顺序的不同,又可以分为三种为:中缀表达式、前缀表达式、后缀表达式

  • 中缀表达式:中缀表达式是一个通用的算术或逻辑公式表示方法,我们从小学就接触的算术表达式就是中缀表达式的写法。比如a + b 就默认是中缀表达式。中缀表达式对大家来说很熟悉,但是对计算机来说比较难计算。因为要比较运算符的优先级,所以一般将中缀表达式转化为后缀表达式再进行表达式的运算。
  • 前缀表达式:前缀表达式是一种没有括号的算术表达式,与中缀表达式不同的是,其将运算符写在前面,操作数写在后面。为纪念其发明者波兰数学家卢卡西维茨(Jan Lukasiewicz),前缀表达式也称为“波兰表达式”。例如,- 1 + 2 3,它等价于1-(2+3)。
  • 后缀表达式:后缀表达式将运算符写在操作数之后。也叫做逆波兰表达式(Reverse Polish notation,RPN,或逆波兰记法)。

在本文中,我们以后缀表达式求值为例,向大家介绍如何利用栈结构计算表达式的值。

首先,我们需要学习后缀表达式运算求值的规则,其运算思路是:从左至右扫描后缀表达式,遇到数字时,将数字压入栈,遇到运算符时,弹出栈顶的两个数,用运算符对它们做出相应的计算,并将结果入栈。重复上述过程,直到表达式的最右端。最后运算得出的值即为表达式的计算结果

文字描述太过抽象,我们通过具体例子进行说明。例如:求后缀表达式3 4 + 5 * 6 -的值,步骤如下:

(1). 从左向右扫描表达式,首先遇到数字,将数字3和数字4压栈。(此时栈中有2个元素为:3、4)

(2). 继续向右扫描,遇到+运算符。弹出4和3(4为栈顶元素,3为次顶元素),计算4+3=7,将结果7压入栈中。(此时,栈中只有1个元素:7)

(3). 向右继续扫描,遇到数字5,将数字5压入栈中。(此时,栈中包含2个元素:7、5)

(4). 向右扫描,遇到号运算符,弹出栈顶元素5和次栈顶元素7,并计算75=35,将结果35压入栈中。(此时,栈中只有1个元素:35)

(5). 向右扫描,遇到数字6,将数字6压入栈中。(此时,栈中包含2个元素:35,6)

(6). 向右扫描,遇到-运算符,弹出栈顶元素6和次栈顶元素35,并计算35-6=29,将数字29压入栈中。(此时,栈中只有1个元素:29)

(7). 整个表达式扫描结束,取出栈中的元素29,就是最后表达式的结果。

二. 队列

1. 队列的概念

队列 (Queue) 也是一种操作受限的线性表,是先进先出的线性表。队列的出口端叫作队头(front),队列的入口端叫作队尾(rear)。队列只允许在队尾进行添加操作,在队头进行删除操作。队列的操作方式和栈类似,唯一的区别在于队列只允许新数据在队尾进行添加,如下图所示:

队列是Java中常用的数据结构,队列的存储结构有两种:一种是基于数组实现的;另一种是基于单链表实现的。

用数组实现队列时,为了入队操作的方便,把队尾位置规定为最后入队元素的下一个位置。 用数组实现的队列叫作顺序队列。数组实现的队列在创建的时候就已经确定了数组的长度,所以队列的长度是固定的,但是可以循环使用数组,所以这种队列也称之为循环队列。

用链表实现的队列叫作链式队列。链表实现的队列内部通过指针指向形成一个队列,这种队列是单向的且长度不固定,所以也称之为非循环队列。

2. 队列的特点

如上图所示,所有的元素都是从队尾进入队列,若需要出队,则从另外一端对头取出元素。我们会发现,元素从队列中出队的顺序刚好是元素进入队列的顺序。我们把这种进入和出来的顺序相同的队列的特点,称之为先进先出(First In First Out,简称为FIFO)

3. 队列的操作

同栈类似,队列也是一种操作受限,队列有入队和出队两种操作。接下来,我们详细介绍:

3.1 入队

入队又称enqueue,入队就是把新元素放入队列中,只允许在队尾的位置放入元素,新元素的下一个位置将会成为新的队尾。添加数据时,首先判断队列的长度是否超出了数组的长度,如果超出则就添加失败(也可以设置成等待,等队列里的数据出队,然后再添加进去)。元素入队完成后,队列长度加一,rear指针也会相应自增一。如下图所示:

3.2 出队

出队又称dequeue,出队就是把元素移出队列,只允许在队头一侧移出元素,出队元素的后一个元素将会成为新的队头。元素出队后,队列的长度减一,front指针自增一。如下图所示:

4. 时间复杂度

无论是顺序队列还是链式队列,入队和出队操作都只涉及1个元素的操作,因此队列操作的时间复杂度都是O(1)。队列的主要应用于资源池、消息队列、命令队列等。

三. 串

1. 概念

我们通常将字符串简称为串,其是由零个或多个字符组成的有限序列。串可以是字母、数字或其他字符。串中的字符数目称为串的长度,零个字符的串称为空串,它的长度为0

串中任意个连续字符组成的子序列称为该串的子串,包含子串的串称为主串。通常我们把字符在序列中的序号为该字符在串中的位置,子串在主串中的位置则以子串的第一个字符在主串中的位置来表示。当两个串的长度相等,并且各个对应的字符都相等时,称两个串相等。由一个或多个空格组成的串称为空格串。空格穿并非空串,它有长度,它的长度为串中空格符号的个数。

2. 串的特性

由此,我们可以得知串的几个结论:

  • 串是由零个或多个字符组成的有限序列
  • 串可以由字母、数字或其他字符组成;
  • 串中的字符数目称为串的长度;
  • 零个字符的串称为空串,它的长度为0;
  • 串中任意个连续的字符组成的子序列,称为该串的子串;
  • 由一个或多个空格组成的串称为空格串;
  • 空格穿并非空串,它有长度,它的长度为串中空格符号的个数。

以上这些简洁的结论,在面试题中可能会遇到,大家需要注意一下。 关于串的其他操作,我们在后面学习算法时,会单独组织一篇内容进行介绍,此处不再赘述。


四. 结语

本篇文章中,向大家介绍了栈、队列和串三种新的线性数据结构。这三种数据结构相对数组和链表而言,操作比较简单,也比较容易理解,各位在学习时需要记住这几个不同数据结构特有的特点。在时间复杂度分析这个指标上,栈和队列的操作均为O(1)。

标签:出栈,怎么回事,队列,元素,栈顶,线性,操作,表达式
From: https://www.cnblogs.com/qian-fen/p/17493157.html

相关文章

  • 数据结构代码整理_队列Queue(C++)
    所谓队列,就是先进先出规则的实现。基于链表实现main.cpp#include<iostream>#include"Queue.h"usingnamespacestd;intmain(){ Queueq; q.append(1); q.append(2); Queue_entrya; q.retrieve(a); cout<<a<<""<<q.empty(); return......
  • 关于线性结构中的双向链表如何实现?
    前言在上一篇文章中,主要是给大家介绍了单向链表的特点及其原理,但是我们没有通过代码进行练习。今天我会继续通过一篇文章,来给大家讲解双向链表的内容,尤其是会通过代码来进行链表的操作,希望大家重点关注哦。全文大约【3500】字,不说废话,只讲可以让你学到技术、明白原理的纯干货!......
  • 延迟队列
    1.延迟队列概念延时队列,队列内部是有序的,最重要的特性就体现在它的延时属性上延时队列中的元素是希望在指定时间到了以后或之前取出和处理简单来说,延时队列就是用来存放需要在指定时间被处理的元素的队列。2.使用场景1.订单在十分钟之内未支付则自动取消2.新创建的店铺,如果......
  • 【题解】Atcoder ABC300 F.More Holidays(线性做法)
    F.MoreHolidays题目描述:给你一个由o和x组成的长度为\(N\)的字符串\(S\),以及整数\(M\)和\(K\)。保证\(S\)至少包含一个x。假设\(T\)是由\(S\)复制\(M\)次而成的长度为\(NM\)的字符串。考虑将\(T\)中的\(K\)个x恰好替换为o。你的目标是在替换后的......
  • 线性基
    线性代数中,我们学过极大线性无关组。极大线性无关组:在线性空间中拥有向量个数最多的线性无关向量组。换言之,任取一个子集所表示的向量不能由集合中剩余的向量表示。在计算机语言中,我们应用在一些方面,称之为线性基。eg.P3812【模板】线性基题意:给你n个数字,取任意个,使它......
  • 随机信号通过线性系统
    冲激响应和传输函数分别为$h(t)$和$H(\omega)$的线性时不变系统,当随机信号输入该线性时不变系统时,其输出的信号是由对应各个输入样本函数的输出响应所构成的函数集合,需要用统计的方法分析输出信号的特征。输入随机信号$X(t)$平稳时,输出响应$Y(t)$1.均值$E[Y(t)]=E[X(t)]H(w)......
  • 数据结构:栈与队列
    栈:栈是一种后进先出的数据结构,我们可以想象为一个瓶子,往里放东西。又比如,函数的递归调用,就是一种栈的结构。php中用数组实现栈:$arr=array();//入栈functionpush(&$arr,$val){$size=count($arr);$arr[$size]=$val;}//出栈functionpop(&$arr){$si......
  • 【题解】CF754D Fedor and coupons(优先队列)
    【题解】CF754DFedorandcoupons题目链接CF754DFedorandcouponsCF1029CMaximalIntersection后者是前者的加强版。思路分析最开始,先考虑不删区间\((k=0)\)的情况:也就是给你一大堆区间,让你找他们的交集。这个还是比较好想的,我们刚开始让第二个区间与第一个区间相交......
  • c++线程安全队列--有锁
    C++线程安全队列是一种数据结构,用于在多线程环境中安全地共享数据。它提供了一组功能,确保多个线程可以同时读取和写入队列,而不会导致竞争条件或数据损坏。C++线程安全队列的常见功能:入队操作(Enqueue):将一个元素添加到队列的尾部。这个操作必须是原子的,以确保在多线程环境中不会......
  • P1903 [国家集训队] 数颜色 / 维护队列 题解
    一、题目描述:给你一个长度为$n$的序列$a$,你需要进行$m$次操作。$类型\1\:将第\x\个元素的值修改为\v\。$$类型\2\:求区间\l\到\r\中有多少种数字。$数据范围:$1\len,m\le1333333,所有数字\le1\times10^6$ 二、解题思路:带......