【數據結構】棧和隊列(上)

目錄

一、棧(先進后出、后進先出的線性表)

1、棧的概念及結構

2、棧的底層結構分析

二、代碼實現

1、定義一個棧

2、棧的初始化

3、入棧

3、增容

4、出棧

5、取棧頂

6、銷毀棧


一、棧(先進后出、后進先出的線性表)

1、棧的概念及結構

棧:一種特殊的線性表,只允許在固定的一端進行插入和刪除數據的操作。進行數據插入、刪除的一端為棧頂,不可進行操作的一端則是棧底。對于任意一個數據結構,我們都要分析一下它的邏輯結構和物理結構,既然是線性表,那么說明邏輯結構一定是線性的

邏輯結構:是線性的

物理結構:也是線性的

對于棧的操作主要有如下兩個:

一個是壓棧,另一個則是出棧。其中壓棧是棧的插入操作(也叫做進棧、入棧);出棧則是棧的刪除操作,出數據也在棧頂。

2、棧的底層結構分析

既然我們已經對棧這一個數據結構的概念有了初步的了解,那么現在,我們來深入探討一下,棧的底層結構是什么?既然他是線性的,那么他是數組還是鏈表呢?

我們從這里分析:

我們可以從上圖看到鏈表的結構,由于棧只能從棧頂操作數據,假設棧的底層是鏈表,如果鏈表的尾部為棧頂的話,每一次訪問棧頂都要去遍歷一次鏈表,那么無疑使時間復雜度增加到了O(N),相反,如果鏈表的頭部為棧頂,對數據進行插入刪除時的時間復雜度就會好很多,為O(1)。

我們來畫個圖看看數組,從數組中插入刪除數據,可以把數組的尾當做棧頂插入刪除數據,時間復雜度認為O(1),既然這樣,鏈表和數組作為棧的底層結構,入棧和出棧的時間復雜度都為O(1),那么,我們來換個維度來考慮:

以上是使用鏈表和數組結構寫的棧的構造代碼,我們可以看出,左圖中的結構使用了鏈表,每向棧中插入一個數據,空間不僅僅會增加int類型的4字節,還會增加指針8字節的大小,相比于數組結構,鏈表結構對空間的使用會更大,所以,我們還是更推薦用數組作為棧的底層結構。

二、代碼實現

下面,我們來實現一下棧的代碼:

1、定義一個棧

我們要定義一下棧的結構,由于棧的底層是數組,又要開辟空間,我們就定義一個動態的數組好了:

typedef int STDataType;
typedef struct Stack
{STDataType* arr;int top;int capacity;
}ST;

2、棧的初始化

下面,我們來定義一個初始化方法

void StackInit(ST* ps)
{ps->arr = NULL;ps->top = ps->capacity = 0;
}

3、入棧

當要向數組插入數據首先要看棧中是否為空,若不為空插入數據后ps->top要++,指向棧頂

3、增容

注意:這里又出現一個問題,當ps->top==ps->capacity時,說明空間已經不夠了,如果要繼續插入數據,我們需要進行一個增容操作:

這里,ps->capacity=newcapacity

void StackPush(ST* ps,STDatatype x)
{assert(ps);//這里做一個斷言操作,防止對空指針解引用if(ps->top == ps->capacity){int newcapacity = ps->capacity==0?4:2*ps->capacity;(STDataType*)tmp = (STDataType*)realloc(ps->arr,newcapacity*sizeof(STDataType));if(tmp == NULL){perror("realloc fail!");exit(1);    }ps->arr = tmp;ps->capacity = newcapacity;}ps->arr[ps->top++] = x;
}

4、出棧

出棧操作:

每當遇到數據結構中的出數據操作時,我們都要考慮其是否有數據可以為我們所用,所以寫出棧操作之前,我們首先要判斷棧是否為空,這里,我們可以定義一個方法:

bool StackEmpty(ST* ps)
{assert(ps);return ps->top == 0;//判斷top(即有效數據個數是否為零,如果為零,則返回ture)
}

其次,假設,ps->top,不為零,則出棧操作直接可以寫成--ps->top;

void Stackpop(ST* ps)
{assert(ps);if(!StackEmpty(ps)){--ps->top;    }
}

5、取棧頂

下一個操作是:取棧頂操作,與出棧頂(有效的數據個數會減少)操作不同,去棧頂操作不會刪除數據,而只是將棧頂的數據復制一下并使用。

STDataType StackTop(ST* ps)
{assert(!StackEmpty(ps);return ps->arr[ps->top-1];
}

6、銷毀棧

接下來,我們看一下對于棧的銷毀操作:

void StackDestroy(ST* ps)
{if(ps->arr)free(ps->arr);ps->arr = NULL;//這里有一點ps->top = ps->capacity = 0;
}

這里有一個小tips:將ps->arr = NULL,這里可能有同學會有疑問,為什么寫出棧操作時,只需要--ps->top,不需要將數據free,這是因為,在數組中,arr表示數組首元素的地址,如果你將ps->arr free掉,那么整個數組的數據都會被銷毀,而鏈表的每一個空間都是獨立存在的,假設你將上一個結點的空間銷毀了,不會影響下一個結點,數組則不然。

以下是測試代碼:

#define  _CRT_SECURE_NO_WARNINGS 1
#include"Stack.h"void test01()
{ST st;StackInit(&st);StackPush(&st, 1);StackPush(&st, 2);StackPush(&st, 3);StackPush(&st, 4);StackPush(&st, 5);/*StackPop(&st);StackPop(&st);StackPop(&st);StackPop(&st);*//*while (!StackEmpty(&st)){int top = StackTop(&st);printf("%d ", top);StackPop(&st);}*/int size = StackSize(&st);printf("size:%d", size);StackDestroy(&st);
}int main()
{test01();return 0;
}

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/pingmian/82099.shtml
繁體地址,請注明出處:http://hk.pswp.cn/pingmian/82099.shtml
英文地址,請注明出處:http://en.pswp.cn/pingmian/82099.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Vue 3 官方 Hooks 的用法與實現原理

Vue 3 引入了 Composition API,使得生命周期鉤子(hooks)在函數式風格中更清晰地表達。本篇文章將從官方 hooks 的使用、實現原理以及自定義 hooks 的結構化思路出發,全面理解 Vue 3 的 hooks 系統。 📘 1. Vue 3 官方生…

大語言模型 17 - MCP Model Context Protocol 介紹對比分析 基本環境配置

MCP 基本介紹 官方地址: https://modelcontextprotocol.io/introduction “MCP 是一種開放協議,旨在標準化應用程序向大型語言模型(LLM)提供上下文的方式。可以把 MCP 想象成 AI 應用程序的 USB-C 接口。就像 USB-C 提供了一種…

云原生安全之PaaS:從基礎到實踐的技術指南

??「炎碼工坊」技術彈藥已裝填! 點擊關注 → 解鎖工業級干貨【工具實測|項目避坑|源碼燃燒指南】 云原生安全之PaaS:從基礎到實踐的技術指南 一、基礎概念 PaaS(Platform as a Service)平臺 PaaS是一種云計算服務模型,為開發者提供應用程序的開發、部署和運行環境,涵…

Chrome中http被強轉成https問題

原因:2023年11月1日,chrome發布HTTPS-Upgrades功能,在用戶訪問 http:// 的舊鏈接之后,會自動嘗試跳轉到通過加密的 https:// 協議,訪問該網站。且探測到 https 服務存在也會自動改成 https。 親測兩種方案可行&#x…

Linux 操作文本文件列數據的常用命令

文章目錄 Linux 操作文本文件列數據的常用命令基本列處理命令高級列處理列數據轉換和排序列數據統計和分析 Linux 操作文本文件列數據的常用命令 Linux 提供了多種強大的命令來處理文本文件中的列數據,以下是一些最常用的命令和工具: 基本列處理命令 c…

如何理解線性判別分析(LDA)算法?

在高維數據空間中,特征變量呈指數級增長,信息分布密集且復雜。研究者在面對海量特征時,仿佛置身于一幅結構高度抽象且維度交織的多變量圖景之中,其解析與建模猶如在一幅復雜的數據宇宙圖譜中導航,既需理論框架的指引,也依賴于算法工具的精確刻畫。如何從眾多維度中篩選出…

鴻蒙UI開發——Builder函數的封裝

1、問題引入 我們在開發中可能會遇到這樣一個問題:將一個Builder修飾后的函數用變量或者數組記錄下來,在業務其他地方使用這些Builder函數。 舉個例子,有下面一段代碼: Builderfunction builderElement() {}let builderArr: Fu…

ARM筆記-ARM指令集

第三章 ARM指令集 3.1 ARM指令集簡介 ARM微處理器的ARM指令集 ,所有的指令長度都是32位 ,并且大多數指令都在一個單獨指令周期內執行。 主要特點: 指令是條件執行的ARM微處理器的指令集是加載/存儲型的在多寄存器操作指令中一次最多可以完成…

Spring Boot接口通用返回值設計與實現最佳實踐

一、核心返回值模型設計(增強版) package com.chat.common;import com.chat.util.I18nUtil; import com.chat.util.TraceUtil; import lombok.AllArgsConstructor; import lombok.Data; import lombok.Getter;import java.io.Serializable;/*** 功能: 通…

2025年上半年軟件架構師考試回憶版【持續更新】

文章目錄 案例分析1、端AI相對于云AI的優勢2、redis持久化,主從庫3、解釋器架構風格4、知識圖譜5、區塊鏈 論文1、基于事件驅動的模型2、多模型數據庫及其應用3、負載均衡設計方法4、論軟件測試理論及其應用 考試感受 2025年軟件考試架構考試于5月24日如期舉行&…

Windows下編譯Zipios

本文記錄在Windows下編譯Zipios的流程。 注1:文章內容會不定期更新。 零、環境 操作系統Windows 11VS Code1.92.1Git2.34.1Visual StudioVisual Studio Community 2022CMake3.22.1 一、安裝依賴 二、編譯 2.1 下載代碼 git clone https://github.com/Zipios/Zi…

SOC-ESP32S3部分:11-任務創建

飛書文檔https://x509p6c8to.feishu.cn/wiki/EH3owsPahisvl6kL6k3cqaQ3n0g 在我們學習單片機的時候,main函數入口中一般有一個while大循環在不停輪詢,如果我們需要實現多種不同的業務,就需要用到狀態機,根據不同時刻的要求執行不…

[Git] 如何進行版本回退

版本控制系統最重要的能力之一,就是能夠輕松地在項目的不同歷史版本之間切換。有時,你可能發現最近的修改引入了嚴重問題,或者需要回到之前的某個節點重新開始。這時,“版本回退”功能就派上用場了。 版本回退:反方向…

易貝平臺關鍵字搜索技術深度解析

一、核心搜索機制 關鍵詞匹配原理 采用TF-IDF算法計算關鍵詞權重 支持同義詞擴展(如"phone"匹配"cellphone") 標題權重 > 副標題 > 商品描述 搜索排序因素 # 搜索權重模擬計算 def calculate_rank(keyword, item): title…

深度剖析 MCP SDK 最新版:Streamable HTTP 模式

好記憶不如爛筆頭,能記下點東西,就記下點,有時間拿出來看看,也會發覺不一樣的感受. 目錄 一、概述 二、快速上手:開啟 Streamable HTTP 服務端開啟 客戶端連接 三、深入兩個核心參數 stateless_http json_resp…

樹莓派開箱上手教程(無需顯示器版)

樹莓派開箱上手教程(無需顯示器版) 硬件準備 名稱參數電源適配器5V電源適配器,至少需要3A的額定電流,配備USB Type-C輸出接頭microSD卡用來將樹莓派的操作系統安裝到上邊,至少需要8GB容量,一般建議16GB及以…

MySQL強化關鍵_015_存儲過程

目 錄 一、概述 1.說明 2.優點 3.缺點 二、存儲過程的操作 1.創建 2.調用 3.查看 4.刪除 三、變量 1.系統變量 (1)說明 (2)查看系統變量 (3)設置系統變量 2.用戶變量 (1&…

動態規劃dp

這里寫目錄標題 動態規劃01背包完全背包多重背包混合背包二維費用的背包分組背包有依賴的背包背包問題求方案數背包問題求具體方案數位 DP狀壓 DP常用例題 動態規劃 01背包 有 n n n 件物品和一個容量為 W W W 的背包,第 i i i 件物品的體積為 w [ i ] w[i] w…

arcgis js統計FeatureLayer的橢球面積、平面面積

1、導入依賴 import FeatureLayer from arcgis/core/layers/FeatureLayer import { geodesicArea, planarArea, simplify } from arcgis/core/geometry/geometryEngine; import { project, load as projectionLoad } from arcgis/core/geometry/projection2、初始化project o…

2.2.1 05年T2

引言 本文將從一預習、二自習、三學習、四復習等四個階段來分析2005年考研英語閱讀第二篇文章。為了便于后續閱讀,我將第四部分復習放在了首位。 四、復習 方法:錯誤思路分析總結考點文章梳理 4.1 錯題分析 題目:26(細節題&…