ImageVerifierCode 换一换
格式:DOCX , 页数:21 ,大小:619.07KB ,
资源ID:3631509      下载积分:3 金币
快捷下载
登录下载
邮箱/手机:
温馨提示:
快捷下载时,用户名和密码都是您填写的邮箱或者手机号,方便查询和重复下载(系统自动生成)。 如填写123,账号就是123,密码也是123。
特别说明:
请自助下载,系统不会自动发送文件的哦; 如果您已付费,想二次下载,请登录后访问:我的下载记录
支付方式: 支付宝    微信支付   
验证码:   换一换

加入VIP,免费下载
 

温馨提示:由于个人手机设置不同,如果发现不能下载,请复制以下地址【https://www.bdocx.com/down/3631509.html】到电脑端继续下载(重复下载不扣费)。

已注册用户请登录:
账号:
密码:
验证码:   换一换
  忘记密码?
三方登录: 微信登录   QQ登录  

下载须知

1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。
2: 试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓。
3: 文件的所有权益归上传用户所有。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 本站仅提供交流平台,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

版权提示 | 免责声明

本文(java数据结构与算法之栈Stack设计与实现.docx)为本站会员(b****5)主动上传,冰豆网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知冰豆网(发送邮件至service@bdocx.com或直接QQ联系客服),我们立即给予删除!

java数据结构与算法之栈Stack设计与实现.docx

1、java数据结构与算法之栈Stack设计与实现java数据结构与算法之栈(Stack)设计与实现栈的抽象数据类型栈是一种用于存储数据的简单数据结构,有点类似链表或者顺序表(统称线性表),栈与线性表的最大区别是数据的存取的操作,我们可以这样认为栈(Stack)是一种特殊的线性表,其插入和删除操作只允许在线性表的一端进行,一般而言,把允许操作的一端称为栈顶(Top),不可操作的一端称为栈底(Bottom),同时把插入元素的操作称为入栈(Push),删除元素的操作称为出栈(Pop)。若栈中没有任何元素,则称为空栈,栈的结构如下图:由图我们可看成栈只能从栈顶存取元素,同时先进入的元素反而是后出,而栈顶

2、永远指向栈内最顶部的元素。到此可以给出栈的正式定义:栈(Stack)是一种有序特殊的线性表,只能在表的一端(称为栈顶,top,总是指向栈顶元素)执行插入和删除操作,最后插入的元素将第一个被删除,因此栈也称为后进先出(Last In First Out,LIFO)或先进后出(First In Last Out FILO)的线性表。栈的基本操作创建栈,判空,入栈,出栈,获取栈顶元素等,注意栈不支持对指定位置进行删除,插入,其接口Stack声明如下:package com.zejian.structures.Stack;/* Created by zejian on 2016/11/27.* 栈接口

3、抽象数据类型*/public interface Stack /* * 栈是否为空 * return */ boolean isEmpty(); /* * data元素入栈 * param data */ void push(T data); /* * 返回栈顶元素,未出栈 * return */ T peek(); /* * 出栈,返回栈顶元素,同时从栈中移除该元素 * return */ T pop();顺序栈的设计与实现顺序栈,顾名思义就是采用顺序表实现的的栈,顺序栈的内部以顺序表为基础,实现对元素的存取操作,当然我们还可以采用内部数组实现顺序栈,在这里我们使用内部数据组来实现栈,至于以

4、顺序表作为基础的栈实现,将以源码提供。这里先声明一个顺序栈其代码如下,实现Stack和Serializable接口:/* * Created by zejian on 2016/11/27. * 顺序栈的实现 */public class SeqStack implements Stack,Serializable private static final long serialVersionUID = -5413303117698554397L; /* * 栈顶指针,-1代表空栈 */ private int top=-1; /* * 容量大小默认为10 */ private int cap

5、acity=10; /* * 存放元素的数组 */ private T array; private int size; public SeqStack(int capacity) array = (T) new Objectcapacity; public SeqStack() array= (T) new Objectthis.capacity; /.省略其他代码代码如下:/* * 获取栈顶元素的值,不删除 * return */ Override public T peek() if(isEmpty() new EmptyStackException(); return arraytop

6、; 从栈添加元素的过程如下(更新栈顶top指向):代码如下:/* * 添加元素,从栈顶(数组尾部)插入 * 容量不足时,需要扩容 * param data */Overridepublic void push(T data) /判断容量是否充足 if(array.length=size) ensureCapacity(size*2+1);/扩容 /从栈顶添加元素 array+top=data;栈弹出栈顶元素的过程如下(删除并获取值):代码如下:/* * 从栈顶(顺序表尾部)删除 * return */ Override public T pop() if(isEmpty() new Empty

7、StackException(); size-; return arraytop-; 到此,顺序栈的主要操作已实现完,是不是发现很简单,确实如此,栈的主要操作就这样,当然我们也可以通过前一篇介绍的MyArrayList作为基础来实现顺序栈,这个也比较简单,后面也会提供带代码,这里就不过多啰嗦了。下面给出顺序栈的整体实现代码:package com.zejian.structures.Stack;import java.io.Serializable;import java.util.EmptyStackException;/* * Created by zejian on 2016/11/27

8、. * Blog : 原文地址,请尊重原创 * 顺序栈的实现 */public class SeqStack implements Stack,Serializable private static final long serialVersionUID = -5413303117698554397L; /* * 栈顶指针,-1代表空栈 */ private int top=-1; /* * 容量大小默认为10 */ private int capacity=10; /* * 存放元素的数组 */ private T array; private int size; public SeqSta

9、ck(int capacity) array = (T) new Objectcapacity; public SeqStack() array= (T) new Objectthis.capacity; public int size() return size; Override public boolean isEmpty() return this.top=-1; /* * 添加元素,从栈顶(数组尾部)插入 * param data */ Override public void push(T data) /判断容量是否充足 if(array.length=size) ensureCa

10、pacity(size*2+1);/扩容 /从栈顶添加元素 array+top=data; size+; /* * 获取栈顶元素的值,不删除 * return */ Override public T peek() if(isEmpty() new EmptyStackException(); return arraytop; /* * 从栈顶(顺序表尾部)删除 * return */ Override public T pop() if(isEmpty() new EmptyStackException(); size-; return arraytop-; /* * 扩容的方法 * par

11、am capacity */ public void ensureCapacity(int capacity) /如果需要拓展的容量比现在数组的容量还小,则无需扩容 if (capacitysize) return; T old = array; array = (T) new Objectcapacity; /复制元素 for (int i=0; isize ; i+) arrayi=oldi; public static void main(String args) SeqStack s=new SeqStack(); s.push(A); s.push(B); s.push(C); Sy

12、stem.out.println(size-+s.size(); int l=s.size();/size 在减少,必须先记录 for (int i=0;i+s.pop(); System.out.println(s.peek-+s.peek(); 链式栈的设计与实现了解完顺序栈,我们接着来看看链式栈,所谓的链式栈(Linked Stack),就是采用链式存储结构的栈,由于我们操作的是栈顶一端,因此这里采用单链表(不带头结点)作为基础,直接实现栈的添加,获取,删除等主要操作即可。其操作过程如下图:从图可以看出,无论是插入还是删除直接操作的是链表头部也就是栈顶元素,因此我们只需要使用不带头结点的

13、单链表即可。代码实现如下,比较简单,不过多分析了:package com.zejian.structures.Stack;import com.zejian.structures.LinkedList.singleLinked.Node;import java.io.Serializable;/* * Created by zejian on 2016/11/27. * Blog : 原文地址,请尊重原创 * 栈的链式实现 */public class LinkedStack implements Stack ,Serializable private static final long se

14、rialVersionUID = 1911829302658328353L; private Node top; private int size; public LinkedStack() this.top=new Node(); public int size() return size; Override public boolean isEmpty() return top=null | top.data=null; Override public void push(T data) if (data=null) throw new StackException(data cant b

15、e null); if(this.top=null)/调用pop()后top可能为null this.top=new Node(data); else if(this.top.data=null) this.top.data=data; else Node p=new Node(data,this.top); top=p;/更新栈顶 size+; Override public T peek() if(isEmpty() throw new EmptyStackException(Stack empty); return top.data; Override public T pop() if

16、(isEmpty() throw new EmptyStackException(Stack empty); T data=top.data; top=top.next; size-; return data; /测试 public static void main(String args) LinkedStack sl=new LinkedStack(); sl.push(A); sl.push(B); sl.push(C); int length=sl.size(); for (int i = 0; i +sl.pop(); 由此可知栈的主要操作都可以在常数时间内完成,这主要是因为栈只对一

17、端进行操作,而且操作的只是栈顶元素。栈的应用栈是一种很重要的数据结构,在计算机中有着很广泛的应用,如下一些操作都应用到了栈。符号匹配中缀表达式转换为后缀表达式计算后缀表达式实现函数的嵌套调用HTML和XML文件中的标签匹配网页浏览器中已访问页面的历史记录接下来我们分别对符合匹配,中缀表达式转换为后缀表达式进行简单的分析,以加深我们对栈的理解。符号匹配 在编写程序的过程中,我们经常会遇到诸如圆括号“()”与花括号“”,这些符号都必须是左右匹配的,这就是我们所说的符合匹配类型,当然符合不仅需要个数相等,而且需要先左后右的依次出现,否则就不符合匹配规则,如“)(”,明显是错误的匹配,而“()”才是正

18、确的匹配。有时候符合如括号还会嵌套出现,如“9-(5+(5+1)”,而嵌套的匹配原则是一个右括号与其前面最近的一个括号匹配,事实上编译器帮我检查语法错误是也是执行一样的匹配原理,而这一系列操作都需要借助栈来完成,接下来我们使用栈来实现括号”()”是否匹配的检测。 判断原则如下(str=”(5-3)*8-2)”):a.设置str是一个表达式字符串,从左到右依次对字符串str中的每个字符char进行语法检测,如果char是,左括号则入栈,如果char是右括号则出栈(有一对匹配就可以去匹配一个左括号,因此可以出栈),若此时出栈的字符char为左括号,则说明这一对括号匹配正常,如果此时栈为空或者出栈字

19、符不为左括号,则表示缺少与char匹配的左括号,即目前不完整。b.重复执行a操作,直到str检测结束,如果此时栈为空,则全部括号匹配,如果栈中还有左括号,是说明缺少右括号。整个检测算法的执行流程如下图:接着我们用栈作为存储容器通过代码来实现这个过程,代码比较简单,如下:package com.zejian.structures.Stack;/* Created by zejian on 2016/11/27.* Blog : 原文地址,请尊重原创* 表达式检测*/public class CheckExpression public static String isValid(String e

20、xpstr) /创建栈 LinkedStack stack = new LinkedStack(); int i=0; while(i中缀表达式(跟日常见到的表达式没啥区别)了解中缀表达式后来看看其定义:将运算符写在两个操作数中间的表达式称为中缀表达式。在中缀表达式中,运算符拥有不同的优先级,同时也可以使用圆括号改变运算次序,由于这两点的存在,使用的中缀表达式的运算规则比较复杂,求值的过程不能从左往右依次计算,当然这也是相对计算机而言罢了,毕竟我们日常生活的计算使用的还是中缀表达式。既然计算机感觉复杂,那么我们就需要把中缀表达式转化成计算机容易计算而且不复杂的表达式,这就是后缀表达式了,在后缀

21、表达式中,运算符是没有优先级的,整个计算都是遵守从左往右的次序依次计算的,如下我们将中缀表达式转为后缀表达式:/1+3*(9-2)+9 转化前的中缀表达式/1 3 9 2 - * + 9 + 转化后的后缀表达式中缀转后缀的转换过程需要用到栈,这里我们假设栈A用于协助转换,并使用数组B用于存放转化后的后缀表达式具体过程如下: 1)如果遇到操作数,我们就直接将其放入数组B中。 2)如果遇到运算符,则我们将其放入到栈A中,遇到左括号时我们也将其放入栈A中。 3)如果遇到一个右括号,则将栈元素弹出,将弹出的运算符输出并存入数组B中直到遇到左括号为止。注意,左括号只弹出并不存入数组。 4)如果遇到任何其

22、他的操作符,如(“+”, “*”,“(”)等,从栈中弹出元素存入数组B直到遇到发现更低优先级的元素(或者栈为空)为止。弹出完这些元素后,才将遇到的操作符压入到栈中。有一点需要注意,只有在遇到” ) “的情况下我们才弹出” ( “,其他情况我们都不会弹出” ( “。 5)如果我们读到了输入的末尾,则将栈中所有元素依次弹出存入到数组B中。 6)到此中缀表达式转化为后缀表达式完成,数组存储的元素顺序就代表转化后的后缀表达式。 执行图示过程如下:简单分析一下流程,当遇到操作数时(规则1),直接存入数组B中,当i=1(规则2)时,此时运算符为+,直接入栈,当i=3(规则2)再遇到运算符*,由于栈内的运算

23、符+优先级比*低,因此直接入栈,当i=4时,遇到运算符(,直接入栈,当i=6时,遇运算符-,直接入栈,当i=8时(规则3),遇),-和(直接出栈,其中运算符-存入后缀数组B中,当i=9时(规则5),由于*优先级比+高,而+与+平级,因此和+出栈,存入数组B,而后面的+再入栈,当i=10(规则5),结束,+直接出栈存入数组B,此时数组B的元素顺序即为1 3 9 2 - * + 9 +,这就是中缀转后缀的过程。 接着转成后缀后,我们来看看计算机如何利用后缀表达式进行结果运算,通过前面的分析可知,后缀表达式是没有括号的,而且计算过程是按照从左到右依次进行的,因此在后缀表达的求值过程中,当遇到运算符时

24、,只需要取前两个操作数直接进行计算即可,而当遇到操作数时不能立即进行求值计算,此时必须先把操作数保存等待获取到运算符时再进行计算,如果存在多个操作数,其运算次序是后出现的操作数先进行运算,也就是后进先运算,因此后缀表达式的计算过程我们也需要借助栈来完成,该栈用于存放操作数,后缀表达式的计算过程及其图解如下:借助栈的程序计算过程:简单分析说明一下: 1)如果ch是数字,先将其转换为整数再入栈 2)如果是运算符,将两个操作数出栈,计算结果再入栈 3)重复1)和2)直到后缀表达式结束,最终栈内的元素即为计算的结果。 整体整体呈现实现如下:package com.zejian.structures.S

25、tack;/* Created by zejian on 2016/11/28.* Blog : 原文地址,请尊重原创* 中缀转后缀,然后计算后缀表达式的值*/public class CalculateExpression /* * 中缀转后缀 * param expstr 中缀表达式字符串 * return */ public static String toPostfix(String expstr) /创建栈,用于存储运算符 SeqStack stack = new SeqStack(expstr.length(); String postfix=;/存储后缀表达式的字符串 int i

26、=0; while (iexpstr.length() char ch=expstr.charAt(i); switch (ch) case +: case -: /当栈不为空或者栈顶元素不是左括号时,直接出栈,因此此时只有可能是*/+-四种运算符(根据规则4),否则入栈 while (!stack.isEmpty() & !stack.peek().equals() postfix += stack.pop(); /入栈 stack.push(ch+); i+; break; case *: case /: /遇到运算符*/ while (!stack.isEmpty() & (stack.peek().equals(*) | stack.peek().equals(/) postfix += stack.pop(); stack.push(ch+); i+; break; case (: /左括号直接入栈 stack.push(ch+); i+; break; case ): /遇到右括号(规则3) String out = stack.p

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1