栈的定义

栈(stack )又称堆栈,它是运算受限的线性表,其限制是仅允许在表的一端进行插入
和删除操作,不允许在其他任何位置进行插入、查找、删除等操作。表中进行插入、删除操
作的一端称为 栈顶(top) ),栈顶保存的元素称为 栈顶元素。相对的,表的另一端称为栈底

当栈中没有数据元素时称为空栈;向一个栈插入元素又称为 进栈或 入栈;从一个栈中删
除元素又称为 出栈或 退栈。由于栈的插入和删除操作仅在栈顶进行,后进栈的元素必定先出
栈,所以又把堆栈称为 后进先出表(Last In First Out,简称 LIFO)。图 4-1 显示了一个堆栈
及数据元素插入和删除的过程。

在这里插入图片描述
当 ABCD 均已入栈之后,出栈时得到的序列为 DCBA,这就是“后进先出”。
在解决实际问题时,如果碰到了数据的使用具有“后进先出”的特性,就预示着可以使用堆
栈来存储和使用这些数据。
堆栈的基本操作除了进栈、出栈操作外,还有判空、取栈顶元素等操作。下面给出堆栈
的抽象数据类型定义。

在这里插入图片描述

栈的顺序存储实现

和线性表类似,堆栈也有两种基本的存储结构:顺序存储结构和链式存储结构。
顺序栈是使用顺序存储结构实现的堆栈,即利用一组地址连续的存储单元依次存放堆栈
中的数据元素。由于堆栈是一种特殊的线性表,因此在线性表的顺序存储结构的基础上,选
择线性表的一端作为栈顶即可。根据数组操作的特性,选择数组下标大的一端,即线性表顺
序存储的表尾来作为栈顶,此时入栈、出栈等操作可以在Ο(1)时间完成。
由于堆栈的操作都在栈顶完成,因此在顺序栈的实现中需要附设一个指针 top 来动态的
指示栈顶元素在数组中的位置。通常 top 可以用栈顶元素所在数组下标来表示,top= -1 时表
示空栈。图 4-1 就可以看成是一个顺序栈。
堆栈在使用过程中所需的最大空间很难估计,因此,一般来说在构造堆栈时不应设定堆
栈的最大容量。一种合理的做法和线性表的实现类似,先为堆栈分配一个基本容量,然后在
实际的使用过程中,当堆栈的空间不够时再倍增存储空间,这个过程所需的时间均摊到每个
数据元素时间为Θ(1),不会影响操作实现的时间复杂度。

public class StackArray implements Stack{
	private Object[] data;
	private int top ; //指针指向数组的下标
	
	public StackArray() {
		top =-1;
		data = new Object[10];
	}

	public Object[] getData() {
		return data;
	}

	public void setData(Object[] data) {
		this.data = data;
	}

	public int getTop() {
		return top;
	}

	public void setTop(int top) {
		this.top = top;
	}

	@Override
	public int getSize() {
		return top+1;
	}

	@Override
	public boolean isEmpty() {
		
		return top<0;
	}
	//入栈
	@Override
	public void push(Object e) {
		group(e);
		data[++top] = e;
		
	}

	private void group(Object e) {
		if((top+1)==data.length){
			Object[] objects = new Object[data.length<<1];
			for(int i=0;i<data.length;i++){
				objects[i] = data[i];
			}
			//重新定义数组
			data = objects;
			
		}
	}
	//出栈
	@Override
	public Object pop() {
		if(isEmpty()){
			throw new RuntimeException("现在没有数据");
		}
		Object pop= data[top--];
		data[top--]=null; //出栈后将改元素下标置为空
		return pop;
	}
	//得到栈顶元素但是不出栈
	@Override
	public Object peek() {
		return data[top];
	}

}

总结:给数组扩容的时候注意重新指向新数组
出栈的时候元素的数组位置应该置为null

栈的链式存储实现

链栈即采用链表作为存储结构实现的栈。当采用单链表存储线性表后,根据单链表的操
作特性选择单链表的头部作为栈顶,此时,入栈、出栈等操作可以在Ο(1)内完成。由于堆
栈的操作只在线性表的一端进行,在这里使用带头结点的单链表或不带头结点的单链表都可
以。使用带头结点的单链表时,结点的插入和删除都在头结点之后进行;使用不带头结点的
单链表时,结点的插入和删除都在链表的首结点上进行。
下面以不带头结点的单链表为例实现堆栈,读者可以对照完成使用带头结点的单链表的
实现。图 4-2 给出了使用不带头结点的单链表实现堆栈的示意图。

在这里插入图片描述

public class StackArrayLinked {
	private DubbloNode head = new DubbloNode(); //伪头节点
	private DubbloNode tail = new DubbloNode(); //伪尾节点
	private int size;

public StackArrayLinked() {
	head.setNext(tail);
	tail.setPre(head);

}

public void push(Object e){
	DubbloNode node = new DubbloNode(e);
	node.setNext(head.getNext());
	node.setPre(head);
	head.setNext(node);
	head.getNext().setPre(node);
	size++;
}

public Object pop(){
	DubbloNode node = head.getNext();
	DubbloNode newnode = node;
	head.setNext(node.getNext());
	return newnode;
}

public int size(){
	return size;
}

}
  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值