基礎算法——鏈表

/*************************************************************************
    > File Name: link_list.cpp
    > Author: xinyang
    > Mail: xuechen.xy@gmail.com 
    > Created Time: Tue 06 Oct 2015 11:42:12 AM CST
 ************************************************************************/

#include <iostream>
#include <stack>
#include <algorithm>
using namespace std;

/*
 * 鏈表節點
 */
struct Node {
    int value;
    Node *next;

    Node(int x) {
        value = x;
        next = NULL;
    }
};

/*
 * 尾插法建立鏈表
 */
Node *insert(Node *head, int x) {
    if (head == NULL) {
        head = new Node(x);
    } else {
        head->next = insert(head->next, x);
    }
    return head;
}

/*
 * 打印鏈表
 */
void print(Node *head) {
    if (head == NULL) {
        return;
    }
    Node *tmp = head;
    while (tmp != NULL) {
        cout << tmp->value << ' ';
        tmp = tmp->next;
    }
}

/*
 * 逆向打印鏈表(迭代)
 */
void print_reverse_iter(Node *head) {
    if (head == NULL) {
        return;
    }

    Node *tmp = head;
    stack<Node *> S;
    while (tmp != NULL) {
        S.push(tmp);
        tmp = tmp->next;
    }

    while (!S.empty()) {
        tmp = S.top();
        S.pop();
        cout << tmp->value << ' ';
    }
}

/*
 * 逆向打印鏈表(遞歸)
 */
void print_reverse_recur(Node *head) {
    if (head == NULL) {
        return;
    }

    if (head != NULL) {
        print_reverse_recur(head->next);
    }
    cout << head->value << ' ';
}

/*
 * O(1)時間內刪除指定節點
 */
void delete_node(Node **head, Node *aim) {
    if (head == NULL || aim == NULL) {
        return;
    }

    if (aim->next != NULL) {
        Node *p = aim->next;
        aim->value = p->value;
        aim->next = p->next;
        delete p;
        p = NULL;
    } else if (*head == aim) {
        delete aim;
        aim = NULL;
        *head = NULL;
    } else {
        Node *tmp = *head;
        while (tmp->next != NULL && tmp->next != aim) {
            tmp = tmp->next;
        }
        tmp->next = NULL;
        delete aim;
        aim = NULL;
    }
}

/*
 * 找出鏈表中倒數第k個節點
 */
Node *find_kth_node(Node *head, int k) {
    if (head == NULL || k <= 0) {
        return NULL;
    }
    Node *fast = head;
    for (int i = 1; i <= k - 1; ++i) {
        if (fast->next == NULL) {
            return NULL;
        }
        fast = fast->next;
    }
    Node *slow = head;
    while (fast->next != NULL) {
        slow = slow->next;
        fast = fast->next;
    }
    return slow;
}

/*
 * 翻轉鏈表
 */
Node *reverse_list(Node *head) {
    if (head == NULL) {
        return NULL;
    }

    Node *ret = NULL, *cur = head, *pre = NULL;
    while (cur != NULL) {
        Node *p = cur->next;
        if (p == NULL) {
            ret = cur;
        }
        cur->next = pre;
        pre = cur;
        cur = p;
    }
    return ret;
}

/*
 * 翻轉m到n之間的鏈表
 */
Node *reverse_between_m_and_n(Node *head, int m, int n) {
    if (head == NULL || m <= 0 || n <= 0 || m >= n) {
        return head;
    }

    Node *pre = NULL, *m_pre = NULL, *cur = head;
    for (int i = 1; i <= n; ++i) {
        if (i == m) {
            m_pre = pre;
        } else if (i > m) {
            pre->next = cur->next;
            cur->next = m_pre->next;
            m_pre->next = cur;
            cur = pre;
        }
        pre = cur;
        cur = cur->next;
    }
    return head;
}

/*
 * 旋轉鏈表,k = 2, 1->2->3->4->5 ==> 4->5->1->2->3
 */
Node *rotate_list(Node *head, int k) {
    if (head == NULL || k <= 0) {
        return head;
    }

    int len = 1;
    Node *tmp = head;
    while (tmp->next != NULL) {
        ++len;
        tmp = tmp->next;
    }

    tmp->next = head;
    k = len - k % len;
    tmp = head;
    for (int i = 1; i < k; ++i) {
        tmp = tmp->next;
    }
    Node *ret = tmp->next;
    tmp->next = NULL;
    return ret;
}

/*
 * 合併兩個排序的鏈表
 */
Node *merge(Node *sorted1, Node *sorted2) {
    if (sorted1 == NULL) {
        return sorted2;
    } else if (sorted2 == NULL) {
        return sorted1;
    }

    Node *ret = NULL;
    if (sorted1->value < sorted2->value) {
        ret= sorted1;
        ret->next = merge(sorted1->next, sorted2);
    } else {
        ret = sorted2;
        ret->next = merge(sorted1, sorted2->next);
    }
    return ret;
}

/*
 * 找出鏈表的第一個公共子節點
 */
int get_length(Node *head) {
    int len = 0;
    Node *tmp = head;
    while (tmp != NULL) {
        ++len;
        tmp = tmp->next;
    }
    return len;
}

Node *find_first_common_node(Node *head1, Node *head2) {
    if (head1 == NULL || head2 == NULL) {
        return NULL;
    }

    int L1 = get_length(head1);
    int L2 = get_length(head2);
    cout << "L1: " << L1 << ' ' << "L2: " << L2 << endl;
    int diff = L1 - L2;
    Node *long_list = head1, *short_list = head2;
    if (diff < 0) {
        diff = L2 - L1;
        long_list = head2;
        short_list = head1;
    }

    for (int i = 0; i < diff; ++i) {
        long_list = long_list->next;
    }

    while (long_list != NULL && short_list != NULL 
            && long_list != short_list) {
        long_list = long_list->next;
        short_list = short_list->next;
    }

    return long_list;
}

/*
 * 找出環鏈表的入口節點
 */
Node *meet_node_within_loop(Node *head) {
    if (head == NULL) {
        return NULL;
    }

    Node *slow = head->next;
    if (slow == NULL) {
        return NULL;
    }

    Node *fast = slow->next;
    while (fast != NULL && slow != NULL) {
        if (fast == slow) {
            return slow;
        }
        slow = slow->next;
        fast = fast->next;
        if (fast != NULL) {
            fast = fast->next;
        }
    }
    return NULL;
}

Node *start_node_of_loop1(Node *head) {
    Node *meet_node = meet_node_within_loop(head);
    if (meet_node == NULL) {
        return NULL;
    }

    int node_in_loop_num = 1;
    Node *p = meet_node;
    while (p->next != meet_node) {
        p = p->next;
        node_in_loop_num++;
    }

    p = head;
    for (int i = 0; i < node_in_loop_num; ++i) {
        p = p->next;
    }

    Node *q = head;
    while (p != q) {
        p = p->next;
        q = q->next;
    }
    return p;
}

Node *start_node_of_loop2(Node *head) {
    if (head == NULL || head->next == NULL) {
        return NULL;
    }

    Node *slow = head, *fast = head;
    while (fast != NULL && fast->next != NULL) {
        slow = slow->next;
        fast = fast->next->next;
        if (slow == fast) {
            Node *slow2 = head;
            while (slow2 != slow) {
                slow2 = slow2->next;
                slow = slow->next;
            }
            return slow2;
        }
    }
    return NULL;
}

/*
 * 刪除鏈表中重複的節點
 */
Node *delete_dup(Node *head) {
    if (head == NULL) {
        return NULL;
    }

    Node *pre = NULL, *cur = head;
    while (cur != NULL) {
        Node *p = cur->next;
        bool flag = false;
        if (p != NULL && p->value == cur->value) {
            flag = true;
        }
        if (flag == false) {
            pre = cur;
            cur = p;
        } else {
            int val = cur->value;
            Node *del = cur;
            while (del != NULL && del->value == val) {
                p = del->next;

                delete del;
                del = NULL;

                del = p;
            }
            if (pre == NULL) {
                head = p;
            } else {
                pre->next = p;
            }
        }
        cur = p;
    }
    return head;
}

/*
 * 去除多餘的重複節點
 */
Node *remove_dup(Node *head) {
    if (head == NULL || head->next == NULL) {
        return NULL;
    }

    Node *pre = head, *cur = head->next;
    while (cur != NULL) {
        if (pre->value == cur->value) {
            pre->next = cur->next;
            delete cur;
        } else {
            pre = cur;
        }
        cur = cur->next;
    }
    return head;
}



int main() {
    Node *head = NULL;
    for (int i = 1; i <= 9; ++i) {
        head = insert(head, i);
    }
    
    cout << "print link list in order" << endl;
    print(head);
    cout << endl << endl;

    cout << "print link list in reverse order" << endl;
    // print_reverse_iter(head);
    print_reverse_recur(head);
    cout << endl << endl;

    /*
    cout << "delete a node in O(1)" << endl;
    Node *aim = head->next->next;
    delete_node(&head, aim);
    print(head);
    cout << endl << endl;
    */

    cout << "find kth node from the end" << endl;
    Node *kth_ret = find_kth_node(head, 3);
    cout << kth_ret->value << endl << endl;

    /*
    cout << "reverse a link list" << endl;
    Node *reverse_ret = reverse_list(head);
    print(reverse_ret);
    cout << endl << endl;
    */

    /*
    cout << "reverse between m and n" << endl;
    Node *reverse_ret = reverse_between_m_and_n(head, 2, 4);
    print(reverse_ret);
    cout << endl << endl;
    */

    cout << "rotate list" << endl;
    Node *rotate_ret = rotate_list(head, 2);
    print(rotate_ret);
    cout << endl << endl;

    cout << "merge two sorted link list" << endl;
    Node *sorted1 = NULL;
    for (int i = 1; i <= 9; i += 2) {
        sorted1 = insert(sorted1, i);
    }
    Node *sorted2 = NULL;
    for (int i = 2; i <= 10; i += 2) {
        sorted2 = insert(sorted2, i);
    }
    Node *merge_ret = merge(sorted1, sorted2);
    print(merge_ret);
    cout << endl << endl;

    cout << "find first common node" << endl;
    Node *second = head->next;
    Node *common_ret = find_first_common_node(head, second);
    cout << common_ret->value << endl << endl;

    cout << "start node of a loop link list" << endl;
    Node *tmp = merge_ret;
    while (tmp->next != NULL) {
        tmp = tmp->next;
    }
    tmp->next = merge_ret->next;
    // Node *start_node = start_node_of_loop1(sorted1);
    Node *start_node = start_node_of_loop2(merge_ret);
    cout << start_node->value << endl << endl;

    cout << "delete duplication nodes" << endl;
    int B[] = {1, 1, 2, 3, 5, 5, 6};
    Node *dup_head = NULL;
    for (int i = 0; i < 7; ++i) {
        dup_head = insert(dup_head, B[i]);
    }
    // Node *dup_ret = delete_dup(dup_head);
    // print(dup_ret);
    Node *dup_ret = remove_dup(dup_head);
    print(dup_ret);
    cout << endl << endl;

    return  0;
}
相關文章
相關標籤/搜索