堆栈的实现

简介: 版权声明:您好,转载请留下本人博客的地址,谢谢 https://blog.csdn.net/hongbochen1223/article/details/45569033 堆栈,是一种数据结构,其插入和删除操作都在同一端进行,其中一端称作栈顶,另外一端称作栈底。
版权声明:您好,转载请留下本人博客的地址,谢谢 https://blog.csdn.net/hongbochen1223/article/details/45569033

堆栈,是一种数据结构,其插入和删除操作都在同一端进行,其中一端称作栈顶,另外一端称作栈底。其中,堆栈是一种先进后出的数据结构,既可以使用公式化描述实现,也可以使用链表描述进行实现。

例如,我们向栈中插入元素10,20.30,则从栈顶向栈底排列分别为30,20,10,则弹出的时候,分别以30,20,10的顺序弹出,在这篇中,先使用公式化描述和链表实现堆栈,在后面将学习几个使用堆栈的比较典型的例子,其中包括括号匹配,汉诺塔,火车重排,离线等价类,迷宫等问题。

好了,话不多说,堆栈的实现还是比较简单的,下面附上我的代码:

首先使用公式化描述实现的堆栈:

/**
 * 堆栈是一个线性表,其插入和删除操作都在同一端进行
 * 其中一端成为栈顶,另一端成为栈底
 *
 * 堆栈是一种先进后出或者是后进先出的数据结构
 * 可以使用公式化描述实现堆栈,也可以使用链表实现堆栈
 *
 */
#include "Exception.h"
#include <iostream>

using namespace std;

/**
 * 适用公式化描述实现LinearStack
 */
template<class T>
class LinearStack{
public:
    LinearStack(int MaxLinearStackSize = 10);
    ~LinearStack(){delete [] stack;}
    bool isEmpty() const{return top == -1;}    //返回栈是否为空
    bool isFull() const{return top == MaxTop;} //栈是否已满
    T Top() const;  //获取栈顶元素
    LinearStack<T>& Add(const T& x);    //向栈中添加一个元素
    LinearStack<T>& Delete(T& x);   //从栈中删除一个元素
    int Size() const;   //获取堆栈的大小
    LinearStack<T>& Add(T array[],int addedSize);  //输入一个堆栈
    void print(); //输出堆栈中的数据

private:
    int top;  //栈顶
    int MaxTop; //栈的最大容量
    T *stack;
};

template <class T>
LinearStack<T>::LinearStack(int MaxLinearStackSize){
    MaxTop = MaxLinearStackSize - 1;
    top = -1;
    stack = new T[MaxLinearStackSize];
}

template<class T>
T LinearStack<T>::Top() const{
    if(isEmpty()) throw OutOfBounds();

    else return stack[top];
}

template<class T>
LinearStack<T>& LinearStack<T>::Add(const T& x){
    if(isFull()) throw OutOfBounds();

    top ++;
    stack[top] = x;

    return *this;
}

template<class T>
LinearStack<T>& LinearStack<T>::Delete(T& x){
    if(isEmpty()) throw OutOfBounds();

    x = stack[top--];

    return *this;
}

template<class T>
int LinearStack<T>::Size() const{

    int size = top+1;

    return size;
}

template<class T>
LinearStack<T>& LinearStack<T>::Add(T array[],int addedSize){
    int size = this->Size();

    int left = MaxTop + 1 - size;

    if(left < addedSize)
        throw OutOfBounds();

    for(int i = 0;i < addedSize;i++){
        this->Add(array[i]);
    }

    return *this;
}

template<class T>
void LinearStack<T>::print(){
    int size = this->Size();

    for(int i = size-1;i >= 0;i--){
        cout << stack[i] << " ";
    }

    cout << endl;
}

下面一个是使用链表实现的堆栈:

/*
 * LinkedStack.cpp
 *
 *  Created on: 2015年5月7日
 *      Author: 洪波
 */

#include "Exception.h"
#include <iostream>

using namespace std;

template<class T>
class LinkedStack;

template<class T>
class Node{
friend class LinkedStack<T>;

private:
    T data;
    Node<T> *link;
};

template<class T>
class LinkedStack{
public:
    LinkedStack(){top = 0;}
    ~LinkedStack();
    bool IsEmpty() const{return top == 0;}
    bool IsFull() const;
    T Top() const;
    LinkedStack<T>& Add(const T& x);
    LinkedStack<T>& Delete(T& x);
    void print();

private:
    Node<T> *top;  //指向栈顶节点
};

template<class T>
LinkedStack<T>::~LinkedStack(){
    Node<T> *next;
    while(top){
        next = top->link;
        delete top;

        top = next;
    }
}

template<class T>
bool LinkedStack<T>::IsFull() const{
    try{
        Node<T> *p = new Node<T>();
        delete p;
        return false;
    }catch(NoMem &nm){
        return true;
    }
}

template<class T>
T LinkedStack<T>::Top() const{

    return top->data;
}

template<class T>
LinkedStack<T>& LinkedStack<T>::Add(const T& x){

    Node<T> *node = new Node<T>();
    node->data = x;
    node->link = top;

    top = node;

    return *this;
}

template<class T>
LinkedStack<T>& LinkedStack<T>::Delete(T& x){
    if(IsEmpty()) throw OutOfBounds();

    Node<T> *p = top;
    x = top->data;
    top = top->link;

    delete p;

    return *this;
}

template<class T>
void LinkedStack<T>::print(){
    Node<T> *p = top;

    while(p){
        cout << p->data << " ";
        p = p->link;
    }

    cout << endl;
}

下面是我的测试类:

/*
 * Test.cpp
 *
 *  Created on: 2015年5月7日
 *      Author: 洪波
 */
#include "LinearStack.cpp"
#include "LinkedStack.cpp"

int main(){

    /**********************************************
     *          测试 LinearStack                    *
     **********************************************/
//  LinearStack<int> ls(10);
//
//  ls.Add(10);
//  ls.Add(20);
//
//
//  ls.print();
//
//  int array[3] = {30,40,50};
//
//  ls.Add(array,3);
//
//  ls.print();

    /*====================================================*/

    /**********************************************
     *          测试 LinkedStack                    *
     **********************************************/

    LinkedStack<int> linkedS;
    linkedS.Add(10);
    linkedS.Add(20);

    linkedS.print();

    int n;
    linkedS.Delete(n);
    cout << n << endl;
    return 0;
}

其中,头文件”Exception.h”在我之前数据结构和算法的代码中有。堆栈在平常使用中用的还是挺多的。但相对来说其实现方式还是比较简单的,其插入和删除操作的时间复杂度都是O(1),即压栈和弹出的时间复杂度都是O(1)。效率还是蛮高的。

向程序运行过程中,函数的调用,在保存状态点的时候,都是保存到栈中的。其中比较典型的应用实例就是火车重排问题,离线等价类问题,汉诺塔问题,迷宫求解问题,括号匹配问题等。我会将在下面一章节中学习这些实例的实现。

目录
相关文章
|
3月前
|
存储 SQL 安全
理解堆栈和内存溢出
【10月更文挑战第05天】
49 3
|
8月前
|
Linux
使用backtrace打印程序crash堆栈
使用backtrace打印程序crash堆栈
109 0
|
存储 算法 C语言
5.堆栈算法
5.堆栈算法
|
存储 Java
堆栈的区别是什么
堆和栈是计算机内存中两种不同的数据结构,它们用来存储程序运行时所需的数据。虽然堆和栈都是用于存储数据的,但它们在内存管理和数据访问方面有着明显的区别。下面我将详细解释堆和栈的区别。
237 0
顺序堆栈和链式堆栈的实现,用一个数组实现两个堆栈的例子
顺序堆栈和链式堆栈的实现,用一个数组实现两个堆栈的例子
|
Java 中间件 Unix
JVM:如何分析线程堆栈
英文原文:JVM: How to analyze Thread Dump 在这篇文章里我将教会你如何分析JVM的线程堆栈以及如何从堆栈信息中找出问题的根因。在我看来线程堆栈分析技术是Java EE产品支持工程师所必须掌握的一门技术。
1725 0
特殊堆栈
数据结构栈的使用
Java多线程-程序运行堆栈分析
class文件内容 class文件包含JAVA程序执行的字节码;数据严格按照格式紧凑排列在class文件中的二进制流,中间无任何分隔符;文件开头有一个0xcafebabe(16进制)特殊的一个标志。 JVM运行时数据区 线程独占:每个线程都会有它独立的空间,随线程生命周期而创建和销毁线程共享:所有线程能访问这块内存数据,随虚拟机或者GC而创建和销毁 方法区 JVM用来存储加载的类信息、常量、静态变量、编译后的代码等数据。
969 0
|
C++ API 数据建模
Windbg查看调用堆栈(k*)
https://www.52pojie.cn/thread-664189-1-1.html       无论是分析程序崩溃原因,还是解决程序hang问题,我们最常查看的就是程序调用堆栈。
1900 0