【數(shù)據(jù)結(jié)構(gòu)】棧和隊(duì)列之鏈棧

該系列文章主要作數(shù)據(jù)結(jié)構(gòu)(C語(yǔ)言版)學(xué)習(xí)的筆記玫坛。里面代碼的注釋很詳細(xì)滋觉,同時(shí)提供給大家參考,歡迎指正和指教昌渤。

棧的鏈?zhǔn)酱鎯?chǔ)結(jié)構(gòu)

棧的鏈?zhǔn)酱鎯?chǔ)結(jié)構(gòu)赴穗,簡(jiǎn)稱為鏈棧

棧因?yàn)橹皇菞m攣?lái)做插入和刪除操作愈涩,所以比較好的方法是將棧頂放在單鏈表的頭部望抽,棧頂指針和單鏈表的指針合二為一。

棧的鏈?zhǔn)酱鎯?chǔ)結(jié)構(gòu)

棧的鏈?zhǔn)酱鎯?chǔ)結(jié)構(gòu)代碼實(shí)現(xiàn)
typedef int SElemType;
// 定義鏈棧
typedef struct StackNode{
    
    SElemType data; // 存放棧的數(shù)據(jù)
    struct StackNode * next;
    
}StackNode, *LinkStackPtr;

typedef struct LinkStack{
    
    LinkStackPtr top; // top指針
    int count; // 棧元素計(jì)數(shù)器
    
}linkStack;

棧的鏈?zhǔn)酱鎯?chǔ)結(jié)構(gòu)操作

一履婉、鏈棧的基本操作:初始化、獲取鏈棧的長(zhǎng)度斟览、判斷是否為空毁腿、獲取棧頂元素等

/**
 * 初始化
 */
Status InitStack(linkStack *S){
    S->top = (LinkStackPtr)malloc(sizeof(StackNode));
    if(!S->top)
        return ERROR;
    S->top=NULL;
    S->count=0;
    return OK;
}

/**
 * 獲取元素
 */
Status visit(SElemType c){
    printf("%d ",c);
    return OK;
}

/**
 * 清空棧
 */
Status ClearStack(linkStack *S){
    LinkStackPtr p,q;
    p=S->top;
    while(p)
    {
        q=p;
        p=p->next;
        free(q);
    }
    S->count=0;
    return OK;
}

/**
 * 判斷棧是否問(wèn)空
 */
Status StackEmpty(linkStack S){
    if (S.count==0)
        return TRUE;
    else
        return FALSE;
}

/**
 * 計(jì)算棧的長(zhǎng)度
 */
int StackLength(linkStack S){
    return S.count;
}
/**
 * 獲取棧頂并返回
 */
Status GetTop(linkStack S,SElemType *e){

    if (S.top == NULL) {
        return ERROR;
    }else{
        *e = S.top->data;
    }
    return OK;
}

二、進(jìn)棧操作(Push)

對(duì)于鏈棧的進(jìn)棧(Push)操作,假設(shè)元素值為e的新節(jié)點(diǎn)是s已烤,top為棧頂指針鸠窗,插入元素e的代碼如下

/**
 * 插入元素 e 為新的棧頂元素
 */
Status Push(linkStack * S, SElemType e){
    
    LinkStackPtr p = (LinkStackPtr)malloc(sizeof(StackNode));
    p->data = e;
    p->next = S->top;   //把當(dāng)前的棧頂元素賦值給新結(jié)點(diǎn)的直接后繼
    S->top = p;        // 將新節(jié)點(diǎn)p賦值給棧頂指針。
    S->count++;
    
    return OK;
}

三胯究、出棧操作(Pop)

至于鏈棧的出棧操作稍计,也是簡(jiǎn)單的三句操作。

  • 假設(shè)變量p用來(lái)存儲(chǔ)要?jiǎng)h除的棧頂結(jié)點(diǎn)裕循。
  • 將棧頂指針下移一位
  • 釋放p臣嚣。
/**
 * 出棧
 */
Status Pop(linkStack * S, SElemType *e){
    
    LinkStackPtr p;
    
    if (StackEmpty(*S)) // 棧為空
        return ERROR;
    
    *e = S->top->data;
    p = S->top;           // 棧頂結(jié)點(diǎn)賦值給p
    S->top = S->top->next; //棧頂指針下移一位
    free(p);
    S->count--;
    return OK;
}

四、遍歷鏈棧操作(Traverse)

/**
 * 遍歷
 */
Status StackTraverse(linkStack S){
    LinkStackPtr p;
    p=S.top;
    while(p)
    {
        visit(p->data);
        p=p->next;
    }
    printf("\n");
    return OK;
}

測(cè)試代碼

int main(int argc, const char * argv[]) {
    @autoreleasepool {

        int j;
        linkStack s;
        int e;
        if (InitStack(&s)) {  //將1到10依次進(jìn)棧
            for (j = 1; j <= 10; j++) {
                Push(&s, j);
            }
        }
        printf("棧中的元素依次為:");
        StackTraverse(s);
        Pop(&s,&e);
        printf("彈出的棧頂元素 e=%d\n",e);
        printf("棸疲空否:%d(1:空 0:否)\n",StackEmpty(s));
        GetTop(s,&e);
        printf("棧頂元素 e=%d 棧的長(zhǎng)度為%d\n",e,StackLength(s));
        ClearStack(&s);
        printf("清空棧后硅则,棧空否:%d(1:空 0:否)\n",StackEmpty(s));
    }
    return 0;
}
測(cè)試結(jié)果

對(duì)比順序棧和鏈棧

  • 時(shí)間復(fù)雜度:均為O(1)
  • 空間性能:
    • 順序棧需要事先確定一個(gè)固定的長(zhǎng)度株婴,可能會(huì)存在內(nèi)存空間浪費(fèi)的問(wèn)題怎虫。但是它的優(yōu)勢(shì)在于存取的時(shí)候很方便。
    • 鏈棧要求每個(gè)元素都有指針域困介,也增加了內(nèi)存開銷大审,但是對(duì)于棧的長(zhǎng)度無(wú)限制。
  • 總體而言:如果棧的使用過(guò)程中元素變化不可預(yù)料座哩,有時(shí)很小徒扶,有時(shí)很大,那么最好用鏈棧八回,反之酷愧,如果變化在可控范圍,使用順序棧更好缠诅。
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末溶浴,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子管引,更是在濱河造成了極大的恐慌士败,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,544評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件褥伴,死亡現(xiàn)場(chǎng)離奇詭異谅将,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)重慢,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,430評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門饥臂,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人似踱,你說(shuō)我怎么就攤上這事隅熙』海” “怎么了?”我有些...
    開封第一講書人閱讀 162,764評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵囚戚,是天一觀的道長(zhǎng)酵熙。 經(jīng)常有香客問(wèn)我,道長(zhǎng)驰坊,這世上最難降的妖魔是什么匾二? 我笑而不...
    開封第一講書人閱讀 58,193評(píng)論 1 292
  • 正文 為了忘掉前任腹泌,我火速辦了婚禮诵次,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘琳轿。我一直安慰自己转培,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,216評(píng)論 6 388
  • 文/花漫 我一把揭開白布浆竭。 她就那樣靜靜地躺著浸须,像睡著了一般。 火紅的嫁衣襯著肌膚如雪邦泄。 梳的紋絲不亂的頭發(fā)上删窒,一...
    開封第一講書人閱讀 51,182評(píng)論 1 299
  • 那天,我揣著相機(jī)與錄音顺囊,去河邊找鬼肌索。 笑死,一個(gè)胖子當(dāng)著我的面吹牛特碳,可吹牛的內(nèi)容都是我干的诚亚。 我是一名探鬼主播,決...
    沈念sama閱讀 40,063評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼午乓,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼站宗!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起益愈,我...
    開封第一講書人閱讀 38,917評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤梢灭,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后蒸其,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體敏释,經(jīng)...
    沈念sama閱讀 45,329評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,543評(píng)論 2 332
  • 正文 我和宋清朗相戀三年摸袁,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了钥顽。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,722評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡靠汁,死狀恐怖耳鸯,靈堂內(nèi)的尸體忽然破棺而出湿蛔,到底是詐尸還是另有隱情膀曾,我是刑警寧澤县爬,帶...
    沈念sama閱讀 35,425評(píng)論 5 343
  • 正文 年R本政府宣布,位于F島的核電站添谊,受9級(jí)特大地震影響财喳,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜斩狱,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,019評(píng)論 3 326
  • 文/蒙蒙 一耳高、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧所踊,春花似錦泌枪、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,671評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至继薛,卻和暖如春修壕,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背遏考。 一陣腳步聲響...
    開封第一講書人閱讀 32,825評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工慈鸠, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人灌具。 一個(gè)月前我還...
    沈念sama閱讀 47,729評(píng)論 2 368
  • 正文 我出身青樓青团,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親咖楣。 傳聞我的和親對(duì)象是個(gè)殘疾皇子督笆,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,614評(píng)論 2 353

推薦閱讀更多精彩內(nèi)容