C/C++数据结构之用链表实现栈

概述

在前一篇文章中,我们已经用数组实现了栈。在本篇文章中,我们将使用链表来实现栈。使用链表的优点是:动态增长,扩容时更加平滑。缺点是:略微复杂,需要额外管理所有节点。栈相关的操作,仍然是下面5个接口。

Push:向栈中添加一个元素。

Pop:从栈中移除顶部元素,并返回该元素。

Top:查看栈顶元素但不移除它。

IsEmpty:检查栈是否为空。

Size:获取栈中元素的数量。

实现原理

我们使用一个带头指针的单链表来实现栈。其中,栈顶对应链表的头部,栈底对应链表的尾部。有一个指针始终指向当前栈顶节点,即链表的第一个节点,每次插入/删除都在链表头部进行。这样做的好处是:所有栈操作只需要修改头指针即可,效率非常高。

在下面栈CLinkedListStack的实现中,我们声明了一个成员变量m_pTop。m_pTop代表栈顶,指向单链表的第一个节点。

cpp 复制代码
class CLinkedListStack
{
public:
    CLinkedListStack();
    ~CLinkedListStack();

    void Push(int nValue);
    int Pop();

    int Top();
    bool IsEmpty();
    int Size();

private:
    // 定义链表节点结构
    struct Node
    {
        int nData;
        struct Node* pNext;
    };

    Node* m_pTop;
};

Push操作

入栈就是在链表的头部插入一个新节点,详细步骤如下。

1、创建一个新节点。

2、新节点的pNext指向当前栈顶。

3、更新栈顶为新节点。

具体如何实现,可参考下面的示例代码。

cpp 复制代码
void CLinkedListStack::Push(int nValue)
{
    Node* pNode = new Node();
    pNode->nData = nValue;
    pNode->pNext = m_pTop;
    m_pTop = pNode;
}

Pop操作

出栈就是删除链表头部节点,并返回其值,详细步骤如下。

1、如果栈为空,抛出异常。

2、取出栈顶节点的值,保存为临时变量。

3、将栈顶指针下移一位。

4、释放旧节点的内存。

5、返回上面的临时变量。

具体如何实现,可参考下面的示例代码。

cpp 复制代码
int CLinkedListStack::Pop()
{
    if (IsEmpty())
    {
        throw underflow_error("stack is empty");
    }

    Node* pTemp = m_pTop;
    int nValue = pTemp->nData;
    m_pTop = m_pTop->pNext;
    delete pTemp;
    return nValue;
}

完整实现

Top操作、IsEmpty操作、Size操作都比较简单,这里就不再赘述了。CLinkedListStack类的完整实现,可参考下面的示例代码。

cpp 复制代码
#include <iostream>
#include <stdexcept>

using namespace std;

CLinkedListStack::CLinkedListStack() : m_pTop(NULL)
{
    NULL;
}

CLinkedListStack::~CLinkedListStack()
{
    while (!IsEmpty())
    {
        Pop();
    }
}

void CLinkedListStack::Push(int nValue)
{
    Node* pNode = new Node();
    pNode->nData = nValue;
    pNode->pNext = m_pTop;
    m_pTop = pNode;
}

int CLinkedListStack::Pop()
{
    if (IsEmpty())
    {
        throw underflow_error("stack is empty");
    }

    Node* pTemp = m_pTop;
    int nValue = pTemp->nData;
    m_pTop = m_pTop->pNext;
    delete pTemp;
    return nValue;
}

int CLinkedListStack::Top()
{
    if (IsEmpty())
    {
        throw underflow_error("stack is empty");
    }

    return m_pTop->nData;
}

bool CLinkedListStack::IsEmpty()
{
    return m_pTop == NULL;
}

int CLinkedListStack::Size()
{
    int nSize = 0;
    Node* pTemp = m_pTop;
    while (pTemp != NULL)
    {
        ++nSize;
        pTemp = pTemp->pNext;
    }

    return nSize;
}

int main()
{
    CLinkedListStack s;
    s.Push(66);
    s.Push(77);
    s.Push(88);

    cout << "Top element: " << s.Top() << endl;
    cout << "Pop: " << s.Pop() << endl;
    cout << "Pop: " << s.Pop() << endl;
    cout << "Current size: " << s.Size() << endl;
    cout << "Is empty: " << (s.IsEmpty() ? "True" : "False") << endl;
    return 0;
}
相关推荐
点云SLAM3 分钟前
C++(C++17/20)最佳工厂写法和SLAM应用综合示例
开发语言·c++·设计模式·c++实战·注册工厂模式·c++大工程系统
Q741_1475 分钟前
C++ 队列 宽度优先搜索 BFS 力扣 662. 二叉树最大宽度 每日一题
c++·算法·leetcode·bfs·宽度优先
Pluchon6 分钟前
硅基计划4.0 算法 动态规划进阶
java·数据结构·算法·动态规划
csdn_aspnet9 分钟前
C++跨平台开发:工程难题与解决方案深度解析
c++
余衫马28 分钟前
在Win10下编译 Poppler
c++·windows·qt·pdf·poppler
王老师青少年编程41 分钟前
2024年3月GESP真题及题解(C++七级): 俄罗斯方块
c++·题解·真题·gesp·csp·俄罗斯方块·七级
oioihoii1 小时前
拆解融合:测试开发,一个关于“更好”的悖论
c++
切糕师学AI1 小时前
ARM 中的 SVC 监管调用(Supervisor Call)
linux·c语言·汇编·arm开发
xiaoqider1 小时前
C++模板进阶
开发语言·c++
移幻漂流1 小时前
C/C++并发编程详解:如何写出优秀的并发程序
c语言·开发语言·c++