首頁 > 軟體

C語言演演算法學習之雙向連結串列詳解

2022-05-14 13:01:32

一、練習題目

題目連結難度
1472. 設計瀏覽器歷史記錄★★★☆☆
430. 扁平化多級雙向連結串列★★★☆☆
劍指 Offer II 028. 展平多級雙向連結串列★★★☆☆
劍指 Offer 36. 二元搜尋樹與雙向連結串列★★★★☆

二、演演算法思路

1、設計瀏覽器歷史記錄

1.這是一個模擬題;

2.初始化生成一個頭結點,記錄一個當前結點;

3.向前 和 向後 是兩個類似的過程,可以統一實現,注意一些邊界條件。

struct Node {
    string val;
    Node* prev;
    Node* next;
};

class BrowserHistory {
    Node * List, *Current;
public:
    BrowserHistory(string homepage) {
        List = new Node();
        List->prev = List->next = nullptr;
        List->val = homepage;

        Current = List;
    }
    
    void visit(string url) {
        Node *Next = Current->next;
        if(Next == nullptr) {
            Current->next = new Node();
            Current->next->next = nullptr;
            Current->next->prev = Current;
        }else {
            Node *tmp = Next->next;
            Next->next = nullptr;
            // free
            while(tmp) {
                Node *node = tmp->next;
                delete tmp;
                tmp = node;
            }
        }
        Current->next->val = url;
        Current = Current->next;
    }
    
    string back(int steps) {
        string str = Current->val;
        Node *pre;
        while(steps-- && Current) {
            pre = Current;
            Current = Current->prev;
            if(Current) str = Current->val;
        }
        if(nullptr == Current) Current = pre;
        return str;

    }
    
    string forward(int steps) {
        string str = Current->val;
        Node *pre;
        while(steps-- && Current) {
            pre = Current;
            Current = Current->next;
            if(Current) str = Current->val;
        }
        if(nullptr == Current) Current = pre;
        return str;
    }
};

2、扁平化多級雙向連結串列

1.利用一個遞迴函數last = dfs(now),一旦遇到child域非空的結點,則遞迴計算clast = dfs(now->child),返回值是遞迴展平後的最後一個結點,然後進行雙向連結串列的連結操作。

2.例如,當前有 child域的結點為now,它的下一個結點是next,遞迴計算以後得到展平的連結串列的最後一個結點為 clast,則有如下關係:

 now <---> now->child    ...    clast <---> next

3.根據以上關係調整雙向連結串列,注意不要忘記將child域置空。

4.當遍歷到這個雙向連結串列的最後一個結點的時候,如果它有child域,則當前連結串列的最後一個結點就是clast,否則就是它自己now;

class Solution {
    Node* dfs(Node* head) {
        Node *now = head;
        Node *last = nullptr;

        while(now) {
            Node *cLast;
            if(now->child) {
                cLast = dfs(now->child);
                Node *next = now->next;

                // now <--> cFirst   ... cLast <---> next;
                now->next = now->child;
                now->child = nullptr;
                now->next->prev = now;

                if(next) {
                    next->prev = cLast; 
                }
                cLast->next = next;
            }
            if(now->next == nullptr) {
                if(now->child) {
                    last = cLast;
                }else {
                    last = now;
                }
            }
            now = now->next;
        }
        return last;
    }
public:
    Node* flatten(Node* head) {
        if(head == nullptr) {
            return nullptr;
        }
        Node *last = dfs(head);
        last->next = nullptr;
        return head;
    }
};

3、展平多級雙向連結串列

(1)同上一題。

4、二元搜尋樹與雙向連結串列

(1)遇到這樣的題,首先需要設計好遞迴函數;

(2)像這個問題,對於 左子樹 和 右子樹,需要知道雙向連結串列的 頭結點 和 尾結點,所以遞迴的時候需要返回 兩個值,於是可以直接採用函數傳指標進行返回,由於二元樹的結點本身就是指標,所以需要傳 二級指標;

(3)遞迴計算左子樹變成雙向連結串列的情況;

(4)遞迴計算右子樹變成雙向連結串列的情況;

(5)將左子樹的雙向連結串列連結到root左邊,將右子樹的雙向連結串列連結到root右邊,然後根據遞迴函數的實際作用,返回 頭結點 和 尾結點。

class Solution {
    void dfs(Node *root, Node **minNode, Node **maxNode) {
        if(root == nullptr) {
            *minNode = nullptr;
            *maxNode = nullptr;
            return ;
        }
        Node *lminNode, *lmaxNode, *rminNode, *rmaxNode;
        if(root->left) {
            dfs(root->left, &lminNode, &lmaxNode);
            lmaxNode->right = root;
            root->left = lmaxNode;
            *minNode = lminNode;
        }else {
            *minNode = root;
        }

        if(root->right) {
            dfs(root->right, &rminNode, &rmaxNode);
            rminNode->left = root;
            root->right = rminNode;
            *maxNode = rmaxNode;
        }else {
            *maxNode = root;
        }
    }
public:
    Node* treeToDoublyList(Node* root) {
        if(root == nullptr) {
            return nullptr;
        }
        Node *minNode, *maxNode;
        dfs(root, &minNode, &maxNode);
        maxNode->right = minNode;
        minNode->left = maxNode;
        return minNode;
    }
};

到此這篇關於C語言演演算法學習之雙向連結串列詳解的文章就介紹到這了,更多相關C語言雙向連結串列內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com