Post

算法(三)--链表

算法(三)--链表

链表

tip:链表旋转操作如果是自己的size整数倍,等于没有操作。所以需要先把旋转次数对size取余。 time %= size

LRU缓存机制 Leetcode 146

使用双向列表和哈希表:

  • 双链表存储一个节点被使用(get或者put)的时间戳,且按最近使用时间从左到右排好序,最先被使用的节点放在双链表的第一位,因此双链表的最后一位就是最久未被使用的节点;
  • 哈希表存储key和对应的节点的地址,通过key可以在O(1)时间内找到对应的节点。

一般希望在O(1)时间内find内存位置的一般需要哈希表,以空间换时间。

注意:双链表的节点结构要保留key值,方便直接根据最右边的节点找到哈希表中对应的key值,然后删除哈希表中的value值。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
class LRUCache {
public:
    // double forward list
    struct Node {
        Node* pre;
        Node* next;
        // key是哈希表的key
        int key_, val_;
        Node(int key, int val)
            : key_(key), val_(val), pre(nullptr), next(nullptr) {};
    } *L, *R;
    int n;
    unordered_map<int, Node*> map;

    // insert
    void insert(Node* node) {
        node->pre = L;
        node->next = L->next;
        L->next->pre = node;
        L->next = node;
    }

    // remove
    void remove(Node* node) {
        node->pre->next = node->next;
        node->next->pre = node->pre;
    }

    LRUCache(int capacity) {
        n = capacity;
        L = new Node(-1, -1), R = new Node(-1, -1);
        L->next = R;
        R->pre = L;
    }

    int get(int key) {
        if (map.count(key) == 0)
            return -1;
        Node* node = map[key];
        remove(node);
        insert(node);
        return node->val_;
    }

    void put(int key, int value) {
        if (map.count(key) > 0) {
            Node* replace = map[key];
            remove(replace);
            Node* node = new Node(key, value);
            map[key] = node;
            insert(node);
        } else {
            if (map.size() == n) {
                Node* old = R->pre;
                remove(old);
                map.erase(old->key_);
                delete old;
            }
            Node* node = new Node(key, value);
            map[key] = node;
            insert(node);
        }
    }
};

快慢指针检测有环链表环开始位置

算法思想:用两个指针,slow每次移动一步,fast每次移动两步,当fast到链表尾时,如果slow追上fast,说明存在环。此时将slow移至表头,快慢指针都同时移动一步,下一次相遇时,就是环产生的地方。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
// 检测环的起始位置
ListNode* detectCycle(ListNode* head) {
    if (!head || !head->next) return nullptr;

    ListNode* slow = head;
    ListNode* fast = head;

    // 检测是否有环
    while (fast && fast->next) {
        slow = slow->next;         // 慢指针移动一步
        fast = fast->next->next;  // 快指针移动两步
        if (slow == fast) {       // 快慢指针相遇,说明有环
            // 找到环的起始位置
            slow = head;
            while (slow != fast) {
                slow = slow->next;
                fast = fast->next;
            }
            return slow; // 返回环的起始节点
        }
    }

    return nullptr; // 如果没有环
}

随机链表的赋值

关键点在于如何通过原来的节点找到新复制的节点,这里用哈希表存储原节点和新节点的对应关系。两遍遍历链表,第一遍遍历原链表,进行深拷贝和建立原节点和新节点的对应关系;第二遍同时遍历原链表和新链表,把原链表的random对应的新节点赋值给新链表的random。

旋转数组

向右移动k个位置,也就是向左移动n-k+1个位置,向左移动比较好处理,只用将头节点移到尾部,不用去找尾节点

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
class Solution {
public:
    ListNode* rotateRight(ListNode* head, int k) {
        if (head == nullptr || head->next == nullptr)
            return head;
        ListNode* dummy = new ListNode(0, head);
        ListNode* cur = head;
        int sum = 0;
        while (cur->next != nullptr) {
            cur = cur->next;
            sum++;
        }
        sum++;
        k %= sum;
        ListNode* last = cur;
        ListNode* newhead = head;
        ListNode* pre = dummy;
        for (int i = 0; i < sum-k; i++) {
            pre = pre->next;
            newhead = newhead->next;
        }

        last->next = dummy->next;
        dummy->next = pre->next;
        pre->next = nullptr;

        return dummy->next;
    }
};

分隔链表

给你一个链表的头节点 head 和一个特定值 x ,请你对链表进行分隔,使得所有 小于 x 的节点都出现在 大于或等于 x 的节点之前。你应当 保留 两个分区中每个节点的初始相对位置。

思想:维护两个链表,一个链表储存小于x的节点,一个链表储存大于等于x的节点,最后将两个链表连接起来。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
    ListNode* partition(ListNode* head, int x) {
        ListNode* small = new ListNode(-1, nullptr);
        ListNode* large = new ListNode(-1, nullptr);
        ListNode* dummys = small;
        ListNode* dummyl = large;

        ListNode* cur = head;
        while(cur) {
            if(cur->val < x) {
                small->next = cur;
                small = small->next;
            }
            else {
                large->next = cur;
                large = large->next;
            }
            cur = cur->next;
        }
        large->next = nullptr;
        small->next = dummyl->next;
        return dummys->next;
    }

千万注意,最后要将large的next置为nullptr,否则会出现环。

This post is licensed under CC BY 4.0 by the author.