單鏈表的相關定義與實現(8.24加入鏈表反轉)

順序表的缺點及解決辦法:

缺點:

  • 插入和刪除時需要移動大量元素,算法時間復雜度為O(n)。
  • 順序線性表長度變化較大時難以確定存儲空間的容量。
  • 造成存儲空間的“碎片”。

解決思路:

  • 所有元素不要考慮相鄰位置了,哪有空位就到哪里,而只是讓每個元素知道它下一個元素的位置在哪里,這樣就可以依次查找了。同時也解決了“難以確定存儲空間容量”的問題了。
  • 思考:這么做是不是也有缺點?
    答:有缺點。如c++標準容器類forward_list用鏈表連接,我要查找里面第n個元素,那么就要從第一個元素開始遍歷,時間復雜度為O(n)。

鏈表方案:

  • 為了表示每個數據元素ai與其直接后繼數據元素ai+1之間的邏輯關系,對數據元素ai來說,除了存儲其本身的信息之外,還需存儲一個指示其直接后繼的信息(即直接后繼的存儲位置)。
    把存儲數據元素信息的域成為數據域,把存儲直接后繼位置的域稱為指針域。這兩部分信息組成數據元素ai的存儲映像,稱為結點(Node)。
  • n個結點鏈結成一個鏈表。即為線性表的鏈式存儲結構。如果每個結點中只包含一個指針域,就稱為單鏈表。
  • 鏈表中第一個結點的存儲位置稱為頭指針。存取從頭指針開始進行。
  • 最后一個指針為尾指針,next指針指向NULL。

項目代碼實現思路:

  • 現在設計一個具體的單鏈表實現。假設項目中,鏈表的數據域存儲一個姓名,指針域存儲下一個姓名。將數據域和指針域用一個struct封裝起來。
  • 要實現的功能為:
    1)獲取單鏈表第i個元素的數據
    2)對單鏈表實現任意位置插入與刪除操作
    3)實現整表創建與整表刪除(C++更好實現,使用構造函數和析構函數即可)
    4)打印鏈表最終內容
1)獲取單鏈表中第i個元素的數據

強調:這種做法需要讓指針遍歷,因此不是一個效率很高的做法。
具體算法思路:

  • 聲明一個指針p指向鏈表第一個結點,初始化j從1開始。
  • 當j<i時,就遍歷鏈表,讓p的指針向后移動,不斷指向下一結點,j累加1。
  • 若到鏈表末尾p為空,則說明第i個結點不存在。
  • 否則查找成功,返回結點p的數據。
    說明:第三條很重要,因為在面試中,一個程序的魯棒性非常重要。它意味著你的程序是否健壯以及是否有抵御bug的能力。
2)在任意位置插入元素

強調:單鏈表的尾插非常方便,但是如果在任意位置插入,則需要遍歷之前的所有元素直至找到當前位置。因此時間復雜度也較高,消耗資源大。
具體算法思路:

  • 聲明一個指針p指向鏈表頭結點,初始化j從1開始。
  • 當j<pos時,就遍歷鏈表,讓p的指針向后移動,不斷指向下一結點,j累加1。
  • 若到鏈表末尾p為空,則說明第pos個結點不存在。(魯棒性)
  • 否則查找成功,在系統中生成一個空結點s。
  • 將string對象st賦值給node->name。
  • 插入標準語句node->next = p->next;,p->next=node。
  • 長度length加一。
  • 返回。
    (p是一個查找用的指針)
3)在任意位置刪除元素

具體算法思路:

  • 核心就是刪除第pos個元素,將第pos-1個指針的next指針繞過第i個元素,指向第pos+1個結點。
  • 首先聲明指針p指向鏈表頭指針,初始化j從1開始。
  • 當j<i時遍歷鏈表,讓p的指針向后移動,不斷指向下一個結點,j累加1。
  • 若到鏈表末尾p為空,則說明第i個結點不存在。
  • 否則查找成功,將欲刪除的結點p->next賦值給q。
  • 單鏈表的刪除標準語句p->next=q->next,p為q之前的結點。
  • 長度length減一。
  • 釋放q結點,返回成功。

具體代碼

1)Linklist.h
#include<string>
using std::string;
struct Node
{
    string name;
    Node * next;
};
class Linklist
{
private:
    Node * head;
    int length;
public:
    Linklist():head(NULL),length(0){};
    ~Linklist();
    Node * GetHead();
    Node * ReverseList(Node * head);
    void Insert(int pos,string st);
    void Delete(int pos);
    void GetLinkListElem(int i,string st);
    void Print();
};
2)Linklist.cpp
#include<iostream>
#include"Linklist.h"
using std::cout;
using std::endl;
Linklist::~Linklist()
{
    Node * temp = new Node;  
    for(int i=0;i<length;i++)  
    {  
        temp = head;  
        head = head->next;  
        delete temp; 
    }
}
Node * Linklist::GetHead()
{
    return this->head;
}
Node * Linklist::ReverseList(Node * head)
{
    Node * node = head;
    Node * prev = NULL;
    while(node != NULL)
    {
        Node * next1 = node->next;
        node->next = prev;
        prev = node;
        node = next1;
    }
    this->head = prev;
    return prev;
}
void Linklist::Insert(int pos,string st)
{
    int j = 1;
    Node * node = new Node;
    Node * p = head;
    if (pos == 1)
    {
        node->name = st;
        node->next = p;
        head = node;
        length++;
        return;
    }
    while(p && j < pos-1)//尋找p==pos-1的位置,在其后插入即為在pos處插入。
    {
        p=p->next;
        j++;
    }
    if(!p || j > pos)//考慮p超出鏈表范圍(變成NULL),pos為0或小于0的情況
    {
        cout << "Cannot insert!" << endl;
        return;
    }
    node->name = st;
    node->next = p->next;
    p->next = node;
    length++;
}
void Linklist::Delete(int pos)
{
    int j = 1;
    Node * p = head;
    if (pos == 1)
    {
        head = head->next;
        length--;
        return;
    }
    while(p && j < pos-1)//尋找p==pos-1的位置,在p->next刪除即為在pos刪除。
    {
        p=p->next;
        j++;
    }
    if(!p || j > pos)//如果p超出范圍或pos太小
    {
        cout << "Cannot delete!" << endl;
        return;
    }
    Node * temp = new Node;
    temp = p->next;
    p->next = temp->next;//把temp后面的結點和p->next鏈起來。
    delete temp;//釋放temp,即p->next。
    length--;
}
void Linklist::Print()
{
    if(head == NULL)
    {
        cout << "Linklist has no elements." << endl;
        return;
    }
    Node * temp = head;
    while(temp != NULL)
    {
        cout << temp->name << "," << endl;
        temp = temp->next;
    }
    cout << endl;
}

3)具體測試main.cpp
#include"Linklist.h"
#include<iostream>
using std::cout;
using std::endl;
int main()
{
    Linklist L;
    L.Insert(1,"sword");
    L.Print();
    L.Insert(2,"edward");
    L.Print();
    L.Insert(3,"ed");
    L.Print();
    Node * head = L.GetHead();
    head = L.ReverseList(head);
    head = L.GetHead();
    L.Print();
    return 0;
}

4)輸出結果

Paste_Image.png
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,646評論 6 533
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,595評論 3 418
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,560評論 0 376
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,035評論 1 314
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,814評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,224評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,301評論 3 442
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,444評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,988評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,804評論 3 355
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,998評論 1 370
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,544評論 5 360
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,237評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,665評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,927評論 1 287
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,706評論 3 393
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,993評論 2 374

推薦閱讀更多精彩內容

  • 鏈表是線性表的鏈式存儲方式,邏輯上相鄰的數據在計算機內的存儲位置不一定相鄰,那么怎么表示邏輯上的相鄰關系呢? 可以...
    rainchxy閱讀 2,008評論 0 6
  • 本文內容取自于小甲魚的數據結構與算法。http://www.lxweimin.com/p/230e6fde9c75 ...
    阿阿阿阿毛閱讀 2,918評論 0 7
  • 1.線性表的定義 線性表:零個或多個數據元素的有限序列序列:也就是說元素之間是有順序的,若元素存在多個,則第一個元...
    e40c669177be閱讀 2,081評論 6 15
  • 大學的時候不好好學習,老師在講臺上講課,自己在以為老師看不到的座位看小說,現在用到了老師講的知識,只能自己看書查資...
    和玨貓閱讀 1,467評論 1 3
  • 完整代碼需結合前面一篇順序表數據結構學習-線性表之順序表各種操作網易云課堂小甲魚課程鏈接:數據結構與算法 線性表的...
    NotFunGuy閱讀 9,234評論 0 9