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

上传人:b****5 文档编号:3631509 上传时间:2022-11-24 格式:DOCX 页数:21 大小:619.07KB
下载 相关 举报
java数据结构与算法之栈Stack设计与实现.docx_第1页
第1页 / 共21页
java数据结构与算法之栈Stack设计与实现.docx_第2页
第2页 / 共21页
java数据结构与算法之栈Stack设计与实现.docx_第3页
第3页 / 共21页
java数据结构与算法之栈Stack设计与实现.docx_第4页
第4页 / 共21页
java数据结构与算法之栈Stack设计与实现.docx_第5页
第5页 / 共21页
点击查看更多>>
下载资源
资源描述

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

《java数据结构与算法之栈Stack设计与实现.docx》由会员分享,可在线阅读,更多相关《java数据结构与算法之栈Stack设计与实现.docx(21页珍藏版)》请在冰豆网上搜索。

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

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

java数据结构与算法之栈(Stack)设计与实现

栈的抽象数据类型

  栈是一种用于存储数据的简单数据结构,有点类似链表或者顺序表(统称线性表),栈与线性表的最大区别是数据的存取的操作,我们可以这样认为栈(Stack)是一种特殊的线性表,其插入和删除操作只允许在线性表的一端进行,一般而言,把允许操作的一端称为栈顶(Top),不可操作的一端称为栈底(Bottom),同时把插入元素的操作称为入栈(Push),删除元素的操作称为出栈(Pop)。

若栈中没有任何元素,则称为空栈,栈的结构如下图:

由图我们可看成栈只能从栈顶存取元素,同时先进入的元素反而是后出,而栈顶永远指向栈内最顶部的元素。

到此可以给出栈的正式定义:

栈(Stack)是一种有序特殊的线性表,只能在表的一端(称为栈顶,top,总是指向栈顶元素)执行插入和删除操作,最后插入的元素将第一个被删除,因此栈也称为后进先出(LastInFirstOut,LIFO)或先进后出(FirstInLastOutFILO)的线性表。

栈的基本操作创建栈,判空,入栈,出栈,获取栈顶元素等,注意栈不支持对指定位置进行删除,插入,其接口Stack声明如下:

packagecom.zejian.structures.Stack;

/**

*Createdbyzejianon2016/11/27.

*栈接口抽象数据类型

*/

publicinterfaceStack{

/**

*栈是否为空

*@return

*/

booleanisEmpty();

/**

*data元素入栈

*@paramdata

*/

voidpush(Tdata);

/**

*返回栈顶元素,未出栈

*@return

*/

Tpeek();

/**

*出栈,返回栈顶元素,同时从栈中移除该元素

*@return

*/

Tpop();

}

顺序栈的设计与实现

  顺序栈,顾名思义就是采用顺序表实现的的栈,顺序栈的内部以顺序表为基础,实现对元素的存取操作,当然我们还可以采用内部数组实现顺序栈,在这里我们使用内部数据组来实现栈,至于以顺序表作为基础的栈实现,将以源码提供。

这里先声明一个顺序栈其代码如下,实现Stack和Serializable接口:

/**

*Createdbyzejianon2016/11/27.

*顺序栈的实现

*/

publicclassSeqStackimplementsStack,Serializable{

privatestaticfinallongserialVersionUID=-5413303117698554397L;

/**

*栈顶指针,-1代表空栈

*/

privateinttop=-1;

/**

*容量大小默认为10

*/

privateintcapacity=10;

/**

*存放元素的数组

*/

privateT[]array;

privateintsize;

publicSeqStack(intcapacity){

array=(T[])newObject[capacity];

}

publicSeqStack(){

array=(T[])newObject[this.capacity];

}

//.......省略其他代码

}

代码如下:

/**

*获取栈顶元素的值,不删除

*@return

*/

@Override

publicTpeek(){

if(isEmpty())

newEmptyStackException();

returnarray[top];

}

从栈添加元素的过程如下(更新栈顶top指向):

代码如下:

/**

*添加元素,从栈顶(数组尾部)插入

*容量不足时,需要扩容

*@paramdata

*/

@Override

publicvoidpush(Tdata){

//判断容量是否充足

if(array.length==size)

ensureCapacity(size*2+1);//扩容

//从栈顶添加元素

array[++top]=data;

}

栈弹出栈顶元素的过程如下(删除并获取值):

代码如下:

/**

*从栈顶(顺序表尾部)删除

*@return

*/

@Override

publicTpop(){

if(isEmpty())

newEmptyStackException();

size--;

returnarray[top--];

}

到此,顺序栈的主要操作已实现完,是不是发现很简单,确实如此,栈的主要操作就这样,当然我们也可以通过前一篇介绍的MyArrayList作为基础来实现顺序栈,这个也比较简单,后面也会提供带代码,这里就不过多啰嗦了。

下面给出顺序栈的整体实现代码:

packagecom.zejian.structures.Stack;

importjava.io.Serializable;

importjava.util.EmptyStackException;

/**

*Createdbyzejianon2016/11/27.

*Blog:

[原文地址,请尊重原创]

*顺序栈的实现

*/

publicclassSeqStackimplementsStack,Serializable{

privatestaticfinallongserialVersionUID=-5413303117698554397L;

/**

*栈顶指针,-1代表空栈

*/

privateinttop=-1;

/**

*容量大小默认为10

*/

privateintcapacity=10;

/**

*存放元素的数组

*/

privateT[]array;

privateintsize;

publicSeqStack(intcapacity){

array=(T[])newObject[capacity];

}

publicSeqStack(){

array=(T[])newObject[this.capacity];

}

publicintsize(){

returnsize;

}

 

@Override

publicbooleanisEmpty(){

returnthis.top==-1;

}

/**

*添加元素,从栈顶(数组尾部)插入

*@paramdata

*/

@Override

publicvoidpush(Tdata){

//判断容量是否充足

if(array.length==size)

ensureCapacity(size*2+1);//扩容

//从栈顶添加元素

array[++top]=data;

size++;

}

/**

*获取栈顶元素的值,不删除

*@return

*/

@Override

publicTpeek(){

if(isEmpty())

newEmptyStackException();

returnarray[top];

}

/**

*从栈顶(顺序表尾部)删除

*@return

*/

@Override

publicTpop(){

if(isEmpty())

newEmptyStackException();

size--;

returnarray[top--];

}

/**

*扩容的方法

*@paramcapacity

*/

publicvoidensureCapacity(intcapacity){

//如果需要拓展的容量比现在数组的容量还小,则无需扩容

if(capacity

return;

T[]old=array;

array=(T[])newObject[capacity];

//复制元素

for(inti=0;i

array[i]=old[i];

}

publicstaticvoidmain(String[]args){

SeqStacks=newSeqStack<>();

s.push("A");

s.push("B");

s.push("C");

System.out.println("size->"+s.size());

intl=s.size();//size在减少,必须先记录

for(inti=0;i

System.out.println("s.pop->"+s.pop());

}

System.out.println("s.peek->"+s.peek());

}

}

链式栈的设计与实现

  了解完顺序栈,我们接着来看看链式栈,所谓的链式栈(LinkedStack),就是采用链式存储结构的栈,由于我们操作的是栈顶一端,因此这里采用单链表(不带头结点)作为基础,直接实现栈的添加,获取,删除等主要操作即可。

其操作过程如下图:

从图可以看出,无论是插入还是删除直接操作的是链表头部也就是栈顶元素,因此我们只需要使用不带头结点的单链表即可。

代码实现如下,比较简单,不过多分析了:

packagecom.zejian.structures.Stack;

importcom.zejian.structures.LinkedList.singleLinked.Node;

importjava.io.Serializable;

/**

*Createdbyzejianon2016/11/27.

*Blog:

[原文地址,请尊重原创]

*栈的链式实现

*/

publicclassLinkedStackimplementsStack,Serializable{

privatestaticfinallongserialVersionUID=1911829302658328353L;

privateNodetop;

privateintsize;

publicLinkedStack(){

this.top=newNode<>();

}

publicintsize(){

returnsize;

}

 

@Override

publicbooleanisEmpty(){

returntop==null||top.data==null;

}

@Override

publicvoidpush(Tdata){

if(data==null){

thrownewStackException("datacan\'tbenull");

}

if(this.top==null){//调用pop()后top可能为null

this.top=newNode<>(data);

}elseif(this.top.data==null){

this.top.data=data;

}else{

Nodep=newNode<>(data,this.top);

top=p;//更新栈顶

}

size++;

}

@Override

publicTpeek(){

if(isEmpty()){

thrownewEmptyStackException("Stackempty");

}

returntop.data;

}

@Override

publicTpop(){

if(isEmpty()){

thrownewEmptyStackException("Stackempty");

}

Tdata=top.data;

top=top.next;

size--;

returndata;

}

//测试

publicstaticvoidmain(String[]args){

LinkedStacksl=newLinkedStack<>();

sl.push("A");

sl.push("B");

sl.push("C");

intlength=sl.size();

for(inti=0;i

System.out.println("sl.pop->"+sl.pop());

}

}

}

由此可知栈的主要操作都可以在常数时间内完成,这主要是因为栈只对一端进行操作,而且操作的只是栈顶元素。

栈的应用

栈是一种很重要的数据结构,在计算机中有着很广泛的应用,如下一些操作都应用到了栈。

符号匹配

中缀表达式转换为后缀表达式

计算后缀表达式

实现函数的嵌套调用

HTML和XML文件中的标签匹配

网页浏览器中已访问页面的历史记录

接下来我们分别对符合匹配,中缀表达式转换为后缀表达式进行简单的分析,以加深我们对栈的理解。

符号匹配

在编写程序的过程中,我们经常会遇到诸如圆括号“()”与花括号“{}”,这些符号都必须是左右匹配的,这就是我们所说的符合匹配类型,当然符合不仅需要个数相等,而且需要先左后右的依次出现,否则就不符合匹配规则,如“)(”,明显是错误的匹配,而“()”才是正确的匹配。

有时候符合如括号还会嵌套出现,如“9-(5+(5+1))”,而嵌套的匹配原则是一个右括号与其前面最近的一个括号匹配,事实上编译器帮我检查语法错误是也是执行一样的匹配原理,而这一系列操作都需要借助栈来完成,接下来我们使用栈来实现括号”()”是否匹配的检测。

判断原则如下(str=”((5-3)*8-2)”):

a.设置str是一个表达式字符串,从左到右依次对字符串str中的每个字符char进行语法检测,如果char是,左括号则入栈,如果char是右括号则出栈(有一对匹配就可以去匹配一个左括号,因此可以出栈),若此时出栈的字符char为左括号,则说明这一对括号匹配正常,如果此时栈为空或者出栈字符不为左括号,则表示缺少与char匹配的左括号,即目前不完整。

b.重复执行a操作,直到str检测结束,如果此时栈为空,则全部括号匹配,如果栈中还有左括号,是说明缺少右括号。

整个检测算法的执行流程如下图:

接着我们用栈作为存储容器通过代码来实现这个过程,代码比较简单,如下:

packagecom.zejian.structures.Stack;

/**

*Createdbyzejianon2016/11/27.

*Blog:

[原文地址,请尊重原创]

*表达式检测

*/

publicclassCheckExpression{

publicstaticStringisValid(Stringexpstr)

{

//创建栈

LinkedStackstack=newLinkedStack<>();

inti=0;

while(i

{

charch=expstr.charAt(i);

i++;

switch(ch)

{

case'(':

stack.push(ch+"");//左括号直接入栈

reak;

case')':

if(stack.isEmpty()||!

stack.pop().equals("("))//遇见右括号左括号直接出栈

return"(";

}

}

//最后检测是否为空,为空则检测通过

if(stack.isEmpty())

return"checkpass!

";

else

return"checkexception!

";

}

publicstaticvoidmain(Stringargs[])

{

Stringexpstr="((5-3)*8-2)";

System.out.println(expstr+""+isValid(expstr));

}

}

中缀表达式转换为后缀表达式

我们先来了解一下什么是中缀表达式,平常所见到的计算表达式都算是中缀表达式,如以下的表达式:

//1+3*(9-2)+9--->中缀表达式(跟日常见到的表达式没啥区别)

了解中缀表达式后来看看其定义:

将运算符写在两个操作数中间的表达式称为中缀表达式。

在中缀表达式中,运算符拥有不同的优先级,同时也可以使用圆括号改变运算次序,由于这两点的存在,使用的中缀表达式的运算规则比较复杂,求值的过程不能从左往右依次计算,当然这也是相对计算机而言罢了,毕竟我们日常生活的计算使用的还是中缀表达式。

既然计算机感觉复杂,那么我们就需要把中缀表达式转化成计算机容易计算而且不复杂的表达式,这就是后缀表达式了,在后缀表达式中,运算符是没有优先级的,整个计算都是遵守从左往右的次序依次计算的,如下我们将中缀表达式转为后缀表达式:

//1+3*(9-2)+9转化前的中缀表达式

//1392-*+9+转化后的后缀表达式

中缀转后缀的转换过程需要用到栈,这里我们假设栈A用于协助转换,并使用数组B用于存放转化后的后缀表达式具体过程如下:

1)如果遇到操作数,我们就直接将其放入数组B中。

2)如果遇到运算符,则我们将其放入到栈A中,遇到左括号时我们也将其放入栈A中。

3)如果遇到一个右括号,则将栈元素弹出,将弹出的运算符输出并存入数组B中直到遇到左括号为止。

注意,左括号只弹出并不存入数组。

4)如果遇到任何其他的操作符,如(“+”,“*”,“(”)等,从栈中弹出元素存入数组B直到遇到发现更低优先级的元素(或者栈为空)为止。

弹出完这些元素后,才将遇到的操作符压入到栈中。

有一点需要注意,只有在遇到”)“的情况下我们才弹出”(“,其他情况我们都不会弹出”(“。

5)如果我们读到了输入的末尾,则将栈中所有元素依次弹出存入到数组B中。

6)到此中缀表达式转化为后缀表达式完成,数组存储的元素顺序就代表转化后的后缀表达式。

执行图示过程如下:

简单分析一下流程,当遇到操作数时(规则1),直接存入数组B中,当i=1(规则2)时,此时运算符为+,直接入栈,当i=3(规则2)再遇到运算符*,由于栈内的运算符+优先级比*低,因此直接入栈,当i=4时,遇到运算符’(‘,直接入栈,当i=6时,遇运算符-,直接入栈,当i=8时(规则3),遇’)’,-和’(‘直接出栈,其中运算符-存入后缀数组B中,当i=9时(规则5),由于*优先级比+高,而+与+平级,因此和+出栈,存入数组B,而后面的+再入栈,当i=10(规则5),结束,+直接出栈存入数组B,此时数组B的元素顺序即为1392-*+9+,这就是中缀转后缀的过程。

接着转成后缀后,我们来看看计算机如何利用后缀表达式进行结果运算,通过前面的分析可知,后缀表达式是没有括号的,而且计算过程是按照从左到右依次进行的,因此在后缀表达的求值过程中,当遇到运算符时,只需要取前两个操作数直接进行计算即可,而当遇到操作数时不能立即进行求值计算,此时必须先把操作数保存等待获取到运算符时再进行计算,如果存在多个操作数,其运算次序是后出现的操作数先进行运算,也就是后进先运算,因此后缀表达式的计算过程我们也需要借助栈来完成,该栈用于存放操作数,后缀表达式的计算过程及其图解如下:

借助栈的程序计算过程:

简单分析说明一下:

1)如果ch是数字,先将其转换为整数再入栈

2)如果是运算符,将两个操作数出栈,计算结果再入栈

3)重复1)和2)直到后缀表达式结束,最终栈内的元素即为计算的结果。

整体整体呈现实现如下:

packagecom.zejian.structures.Stack;

/**

*Createdbyzejianon2016/11/28.

*Blog:

[原文地址,请尊重原创]

*中缀转后缀,然后计算后缀表达式的值

*/

publicclassCalculateExpression{

/**

*中缀转后缀

*@paramexpstr中缀表达式字符串

*@return

*/

publicstaticStringtoPostfix(Stringexpstr)

{

//创建栈,用于存储运算符

SeqStackstack=newSeqStack<>(expstr.length());

Stringpostfix="";//存储后缀表达式的字符串

inti=0;

while(i

{

charch=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)

Stringout=stack.p

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > 小学教育 > 小升初

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

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