算法

《剑指offer》链表专题

2018-02-01  本文已影响69人  wenmingxing

链表

记录《剑指offer》中所有关于链表的题目,以及LeetCode中的相似题目

相关题目列表

index description key words done date
5 从尾到头打印链表 栈的应用 Y 18-2-1
13 在O(1)时间删除链表结点 删除结点 Y 18-2-1
15 链表中倒数第k个结点 双指针 Y 18-2-1
16 反转链表 反转 Y 18-2-2
17 合并两个排序的链表 递归合并 Y 18-2-2
26 复杂链表的复制 链表复制 Y 18-2-2
27 二叉搜索树与双向链表 与链表的关系 Y 18-4-2
37 两个链表的第一个公共结点 快慢指针 Y 18-2-3
45 圆圈中最后剩下的数字 约瑟夫环 Y 18-2-14
56 链表中环的入口结点 快慢指针 Y 18-2-3
57 删除链表中重复的结点 删除节点 Y 18-2-14

题目

链表是面试中最常出现的题目之一,其特点是包含大量的指针操作,因此这对被面试者对指针的理解是否深入要求很高。

面试题5: 从尾到头打印链表

题目:输入一个链表的头结点,从尾到头反过来打印出每个节点的值。

题目分析

本题最暴力的方式是逐次遍历链表输出对应节点的值。但明显这种方式的时间复杂度是我们无法接受的。有没有高效的方法呢。
其实很容易想到可以利用栈的先入后出的特性。遍历链表,并将遍历到的元素依次push进栈,然后,只需直接这个pop即为题目要求的从尾到头打印链表。

参考代码

#include<iostream>
#include<stack>
#include<vector>


using namespace std;

//创建单链表节点
struct ListNode
{
    int m_nValue;
    ListNode* m_pNext;
};

/*=======================在单向链表尾增添元素==========================*/

//考虑如果链表本身为空的时候,添加元素则新添加元素赋给pHead,
//如果不用二重指针,则在函数外不能改变*pHead的指向
void AddToTail(ListNode** pHead, int value)
{
    ListNode* pNew = new ListNode();    //使用new创建对象
    pNew->m_nValue = value;
    pNew->m_pNext = NULL;

    if (*pHead == NULL)     //如果链表为空,则直接赋值
        *pHead = pNew;
    else
    {
        ListNode* pNode = *pHead;   //链表不为空,则通过移动指针pNode至最后
        while (pNode->m_pNext != NULL)
        {
            pNode = pNode->m_pNext;
        }

        pNode->m_pNext = pNew;  //添加元素
    }
}

/*=====================删除给定值节点=================================*/
void RemoveNode(ListNode** pHead, int value)
{
    if (*pHead == NULL || pHead == NULL)      //
        return;

    ListNode* pToBeDeleted = NULL;  //临时节点

    if ((*pHead)->m_nValue == value)
    {
        pToBeDeleted = *pHead;
        *pHead = (*pHead)->m_pNext;
    }
    else
    {
        ListNode* pNode = *pHead;
        while (pNode->m_pNext != NULL && pNode->m_pNext->m_nValue != value)
            pNode = pNode->m_pNext;

        if (pNode->m_pNext != NULL && pNode->m_pNext->m_nValue == value)
        {
            pToBeDeleted = pNode->m_pNext;
            pNode->m_pNext = pNode->m_pNext->m_pNext;
        }
    }

    if (pToBeDeleted != NULL)
    {
        delete pToBeDeleted;     //释放内存
        pToBeDeleted = NULL;     //释放指针
    }
}

/*=========================测试类与main函数======================*/

/*
class Solution
{
public:
    vector<int> res;
    vector<int> PrintListFromTailToHead(ListNode *pHead)
    {
        if (pHead != NULL)
        {
            if (pHead->m_pNext != NULL)
            {
                PrintListFromTailToHead(pHead->m_pNext);
            }
            res.push_back(pHead->m_nValue);
        }
        return res;
    }
};
int main()
{
    ListNode list[4];
    list[0].m_nValue = 1;
    list[0].m_pNext = &list[1];
    list[1].m_nValue = 2;
    list[1].m_pNext = &list[2];
    list[2].m_nValue = 3;
    list[2].m_pNext = &list[3];
    list[3].m_nValue = 4;
    list[3].m_pNext = NULL;
    ListNode *node = *head;
    Solution result;
    vector<int> res = result.PrintListFromTailToHead(node);
    for (int i = 0; i < res.size(); ++i)
    {
        cout << res[i] << endl;
    }
    return 0;
}
*/


/*===========================从尾到头打印链表========================*/
void PrintListfromTailtoHead(ListNode* pHead)
{
    std::stack<ListNode*> nodes;

    ListNode* pNode = pHead;
    while (pNode != NULL)
    {
        nodes.push(pNode);
        pNode = pNode->m_pNext;
    }

    while (!nodes.empty())
    {
        pNode = nodes.top();
        cout << pNode->m_nValue << " ";
        nodes.pop();
    }
}

/*======================递归方式=====================*/
void PrintListfromTailtoHead_Recursively(ListNode* pHead)
{
    if (pHead != NULL)
    {
        if (pHead->m_pNext != NULL)
        {
            PrintListfromTailtoHead_Recursively(pHead->m_pNext);
        }
        cout << pHead->m_nValue << " ";
    }
}

相似题目

可以在牛客网 剑指offer上完成对本题的测试。

面试题13: 在O(1)时间删除链表节点

题目: 给定单向链表的头指针和一个结点指针,定义一个函数在O(1)时间删除该节点。

题目分析

上题中的参考代码中给出了一般情况下删除链表节点的代码。但是由于链表的特殊性,我们首先需要遍历链表找到该节点的上一个节点才能实施删除操作。
这是由于我们要得到删除节点的前一个节点才能对链表进行有效的删除操作。那么本题的关键就在于是否存在一种方式可以不需要得到前一个节点而完成删除操作呢?
如果我们把下一个节点的内容复制到需要删除节点上覆盖原来的内容,再把下一个节点删除,那就相当于完成了对节点的删除操作。也就是说实际上我们是通过将该节点的下一节点删除而替代了本节点的删除。
但是,这样做也存在一个问题,如果要删除节点的下一个节点不存在怎么办,也就是要删除节点本身是尾结点,这样我们只能通过一般的删除操作完成。就是参考代码如下:

参考代码

#include<iostream>

using namespace std;

struct ListNode
{
    int m_nValue;
    ListNode* m_pNext;
};

void DeleteNode(ListNode** pListHead, ListNode* pToBeDeleted)
{
    if (!pListHead || !pToBeDeleted)
        return;

    //要删除的结点不是尾结点
    if (pToBeDeleted->m_pNext != NULL)
    {
        ListNode* pNext = pToBeDeleted->m_pNext;
        pToBeDeleted->m_nValue = pNext->m_nValue;
        pToBeDeleted->m_pNext = pNext->m_pNext;

        delete pNext;
        pNext = NULL;
    }

    //只有一个结点,删除头结点(也是尾结点)
    else if (*pListHead == pToBeDeleted)
    {
        delete pToBeDeleted;
        pToBeDeleted = NULL;
        *pListHead = NULL;
    }

    //链表有多个结点,删除尾结点
    else
    {
        ListNode* pNode = *pListHead;
        while (pNode->m_pNext != pToBeDeleted)
        {
            pNode = pNode->m_pNext;
        }
        pNode->m_pNext = NULL;
        delete pToBeDeleted;
        pToBeDeleted = NULL;
    }
}

相似题目

本题与LeetCode中的237. Delete Node in a Linked List类似,另外LeetCode中还有一道删除所有节点元素值为某一指定值的所有节点的题目203. Remove Linked List Elements
这两道题的参考代码见:
LeetCode 237 code
LeetCode 203 code

面试题15: 链表中倒数第k个节点

题目: 输入一个链表,输出该链表中倒数第k个节点。为了符合大多数人的习惯,本题从1开始计数,即链表的尾结点是倒数第1个节点。

题目分析

本题可以先遍历链表的到链表长度,从而可以确定倒数第k个节点的正向位置,从而再通过一次遍历得出题解,下面的代码中采用的双指针滑动其实也与这种方法本质一样,但需要注意算法的判定条件以保证其鲁棒性。

参考代码

#include<iostream>

using namespace std;

struct ListNode
{
    int m_nValue;
    ListNode* m_pNext;
};

ListNode* FindKthToTail(ListNode* pListHead, int k)
{
    if (pListHead == NULL)  //如果输入链表
        return NULL;
    if (k == 0)     //如果k==0
        return NULL;

    ListNode* pAhead = pListHead;
    ListNode* pBehind = pListHead;

    for (int i = 0; i < k - 1; ++i)
    {
        if (pAhead->m_pNext != NULL)      //判断不能让输入的k值小于链表长度
        {
            pAhead = pAhead->m_pNext;
        }
        else       //证明k超出了链表长度
        {
            return NULL;
        }
    }

    while (pAhead->m_pNext != NULL)
    {
        pAhead = pAhead->m_pNext;
        pBehind = pBehind->m_pNext;
    }
    return pBehind;
}

int main()
{
    //->运算符需要前面是指针(指向类对象的指针),.运算符需要前面是类的对象。
    ListNode List[5];
    List[0].m_nValue = 1;
    List[0].m_pNext = &List[1];
    List[1].m_nValue = 2;
    List[1].m_pNext = &List[2];
    List[2].m_nValue = 3;
    List[2].m_pNext = &List[3];
    List[3].m_nValue = 4;
    List[3].m_pNext = &List[4];
    List[4].m_nValue = 5;
    List[4].m_pNext = NULL;

    cout << FindKthToTail(List, 2)->m_nValue << endl;

    return 0;
}

相似题目

本题包含于LeetCode中的19. Remove Nth Node From End of List,LeetCode中的题目增加了对删除该节点的要求,参考代码见:
LeetCode 19 code
还可以在牛客网 剑指offer上完成对本题的练习。

面试题16: 反转链表

题目: 定义一个函数,输入一个链表的头结点,反转该链表并输出反转后链表的头结点。

题目分析

要实现一个链表的翻转首先需要得到一个节点的前后节点,所以本题采用三指针滑动方法。这就涉及了大量的指针操作。

参考代码

//三指针滑动法
#include<iostream>

using namespace std;

struct ListNode
{
    int m_nValue;
    ListNode* m_pNext;
};

ListNode* ReversetList(ListNode* pHead)
{
    ListNode* pPrev = NULL;
    ListNode* pNode = pHead;
    ListNode* pNewHead = NULL;

    while (pNode != NULL)
    {
        ListNode *pNext = pNode->m_pNext;

        if (pNext == NULL)  //到达最后一个赋给新链表头部
            pNewHead = pNode;

        pNode->m_pNext = pPrev;
        pPrev = pNode;
        pNode = pNext;
    }
    return pNewHead;
}

int main()
{
    ListNode List[5];

    List[0].m_nValue = 1;
    List[0].m_pNext = &List[1];
    List[1].m_nValue = 2;
    List[1].m_pNext = &List[2];
    List[2].m_nValue = 3;
    List[2].m_pNext = &List[3];
    List[3].m_nValue = 4;
    List[3].m_pNext = &List[4];
    List[4].m_nValue = 5;
    List[4].m_pNext = NULL;

    ListNode* node = List;

    while (node != NULL)
    {
        cout << node->m_nValue << " ";
        node = node->m_pNext;
    }

    cout << endl;

    node = ReversetList(List);  //链表必须以头结点开始才能遍历,所以要返回一个node

    while (node != NULL)
    {
        cout << node->m_nValue << " ";
        node = node->m_pNext;
    }

}

相似题目

本题与LeetCode中的206. Reverse Linked List完全一致;另外LeetCode中还有一道本题的引申92. Reverse Linked List II为翻转链表的局部。
这两题的参考代码见:
LeetCode 206 code
LeetCode 92 code
还可以在牛客网 剑指offer上完成对本题的练习。

面试题17: 合并两个排序的链表

题目: 输入两个递增排序的链表,合并这两个链表并使新链表中的节点仍然按照递增排序。

题目分析

合并两个链表的过程可以看成是一个不断按照顺序添加元素形成一个链表的过程。可以以递归的方式解决。

参考代码

#include<iostream>

using namespace std;

struct ListNode
{
    int val;
    ListNode* next;
};

class Solution
{
public:
    ListNode* Merge(ListNode* pHead1, ListNode* pHead2)
    {
        if (pHead1 == NULL)
            return pHead2;
        if (pHead2 == NULL)
            return pHead1;

        ListNode* pNewHead = NULL;

        if (pHead1->val <= pHead2->val)
        {
            pNewHead = pHead1;
            pNewHead->next = Merge(pHead1->next, pHead2);
        }
        if (pHead1->val > pHead2->val)
        {
            pNewHead = pHead2;
            pNewHead->next = Merge(pHead1, pHead2->next);
        }
        return pNewHead;
    }
};

int main()
{
    ListNode List1[3];
    ListNode List2[3];

    List1[0].val = 1;
    List1[0].next = &List1[1];
    List1[1].val = 3;
    List1[1].next = &List1[2];
    List1[2].val = 5;
    List1[2].next = NULL;

    List2[0].val = 2;
    List2[0].next = &List2[1];
    List2[1].val = 4;
    List2[1].next = &List2[2];
    List2[2].val = 6;
    List2[2].next = NULL;

    Solution solu;

    ListNode* node = solu.Merge(List1, List2);

    while (node != NULL)
    {
        cout << node->val << " ";
        node = node->next;
    }
    return 0;
}

相似题目

本题与LeetCode中的21. Merge Two Sorted Lists完全一致;另外LeetCode中还有一道本题的引申,即为合并k个排序链表23. Merge k Sorted Lists。这两题的参考代码见:
LeetCode 21 code
LeetCode 23 code
还可以在牛客网 剑指offer上完成对本题的练习。

面试题26: 复杂链表的复制

题目: 请实现函数,复制一个复杂链表。在复杂链表中,每个节点除了有一个next指针指向下一个节点外,还有一个指针指向链表中的任意节点或者nullptr。

题目分析

本题可以采用三个步骤完成:
不妨设原始链表为N,新链表为N'。
第一步,这个节点复制N,并将N'放在每个复制节点后面。

第一步

第二步,复制复杂指针。N'对应节点的复杂指针应该在第一步得到链表上N的next。

第二步

第三步,拆分链表。

第三步

参考代码

/*
struct RandomListNode {
    int label;
    struct RandomListNode *next, *random;
    RandomListNode(int x) :
            label(x), next(NULL), random(NULL) {
    }
};
*/
class Solution {
public:
    RandomListNode* Clone(RandomListNode* pHead)
    {
        CloneNodes(pHead);
        ConnectRandomNodes(pHead);
        return ReconnectNodes(pHead);
    }
private:
    //复制节点
    void CloneNodes(RandomListNode* pHead){
        RandomListNode* pNode = pHead;
        while (pNode != NULL){
            RandomListNode* pCloned = new RandomListNode(pNode->label);     //每次必须新建节点
            pCloned->label = pNode->label;
            pCloned->next = pNode->next;
            pCloned->random = NULL;

            pNode->next = pCloned;
            pNode = pCloned->next;
        }
    }

    //设置random指针
    void ConnectRandomNodes(RandomListNode* pHead){
        RandomListNode* pNode = pHead;
        while (pNode != NULL){
            RandomListNode* pCloned = pNode->next;
            if (pNode->random != NULL)
                pCloned->random = pNode->random->next;
            pNode = pCloned->next;
        }
    }

    //拆分链表
    RandomListNode* ReconnectNodes(RandomListNode* pHead){
        RandomListNode* pNode = pHead;
        RandomListNode* pClonedHead = NULL;
        RandomListNode* pClonedNode = NULL;

        if (pNode != NULL){     //处理头结点
            pClonedHead = pClonedNode = pNode->next;
            pNode->next = pClonedNode->next;
            pNode = pNode->next;
        }
        while (pNode != NULL){
            pClonedNode->next = pNode->next;
            pClonedNode = pClonedNode->next;
            pNode->next = pClonedNode->next;
            pNode = pNode->next;
        }
        return pClonedHead;
    }
};

相似题目

本题与LeetCode中的138. Copy List with Random Pointer相似。参考代码见:
LeetCode 138 code
还可以在牛客网 剑指offer上完成对本题的练习。

面试题27: 二叉树与双向链表

题目: 输入一棵二叉搜索树,将该二叉搜索树转换成一个排序的双向链表。要求不能创建任何新的结点,只能调整树中结点指针的指向。

题目分析

根据BST与排序双向链表的关系,原先指向左子结点的指针调整为链表中指向前一个结点的指针,原先指向右子结点的指针调整为链表中指向后一个结点的指针。

接下来我们考虑如何进行转换。

根据BST中序遍历有序的特点,我们采用中序遍历算法从小到大遍历二叉树的每一个结点。当遍历到根结点时,我们把树看成3部分(如下图所示),值为10的结点、根结点值为6的左子树,根结点值为14的右子树。根据排序链表的定义,值为10的节点将和它的左子树的最大一个节点相连,同时与右子树中最小节点相连。

根据中序遍历的顺序,当我们遍历转换到根结点时,它的左子树已经转换成一个排序链表了,并且最后一个节点即为其中最大的结点。我们把8与10相连即可。接着遍历转换右子树,并把根结点和右子树中最小的结点相连。

参考代码

/*
struct TreeNode {
    int val;
    struct TreeNode *left;
    struct TreeNode *right;
    TreeNode(int x) :
            val(x), left(NULL), right(NULL) {
    }
};*/
class Solution {
public:
    TreeNode* Convert(TreeNode* pRootOfTree)
    {
        TreeNode* pLastNodeInList = NULL;
        ConverNode(pRootOfTree, &pLastNodeInList);  //因为函数形参是二重指针,所以需要用取地址符

        TreeNode* pHeadOfList = pLastNodeInList;
        while (pHeadOfList != NULL && pHeadOfList->left != NULL)
            pHeadOfList = pHeadOfList->left;
        return pHeadOfList;
    }
private:
    //类似于中序遍历
    void ConverNode(TreeNode* pNode, TreeNode** pLastNodeInList){   //因为需要对pLastNodeInList进行动态改变,所以需要用二重指针
        if (pNode == NULL)
            return;

        TreeNode* pCurrent = pNode;
        if (pCurrent->left != NULL)
            ConverNode(pCurrent->left, pLastNodeInList);

        pCurrent->left = *pLastNodeInList;

        if (*pLastNodeInList != NULL)
            (*pLastNodeInList)->right = pCurrent;

        *pLastNodeInList = pCurrent;

        if (pCurrent->right != NULL)
            ConverNode(pCurrent->right, pLastNodeInList);
    }
};

相似题目

本题与LeetCode中的109. Convert Sorted List to Binary Search Tree类似,其是将单链表转化为BST。
还可以在牛客网 剑指offer上完成对本题的练习。

面试题37: 两个链表的第一个公共结点

题目: 输入两个链表,找出它们的第一个公共结点。

题目分析

本题是典型的快慢指针应用问题,要求得两个链表的公共节点可以先得到两个链表的长度,则通过长度可以进行快慢指针的设置,例如假如l1的长度为m,l2的长度为n,且不妨设m>n,于是可以设置在较长的链表l1上先走m-n步;在同时便利,则到达第一个相同的节点即为第一个公共节点。

参考代码

/*
struct ListNode {
    int val;
    struct ListNode *next;
    ListNode(int x) :
            val(x), next(NULL) {
    }
};*/
class Solution {
public:
    ListNode* FindFirstCommonNode( ListNode* pHead1, ListNode* pHead2) {
        if (pHead1 == NULL || pHead2 == NULL)
            return NULL;
        int pHead1_size = 0;
        int pHead2_size = 0;

        ListNode* pNode1 = pHead1;
        ListNode* pNode2 = pHead2;
        while (pNode1 != NULL){     //得到pHead1的size
            pHead1_size++;
            pNode1 = pNode1->next;
        }
        while (pNode2 != NULL){     //得到pHead2的size
            pHead2_size++;
            pNode2 = pNode2->next;
        }

        int length = pHead1_size - pHead2_size;
        //cout << length << endl;
        if (length > 0){
            pNode2 = pHead2;
            pNode1 = pHead1;
            while (length != 0){
                pNode1 = pNode1->next;
                length--;
            }
        }
        else if (length < 0){
            pNode1 = pHead1;
            pNode2 = pHead2;
            while (length != 0){
                pNode2 = pNode2->next;
                length++;
            }
        }
        else if (length == 0){
            pNode1 = pHead1;
            pNode2 = pHead2;
        }

        while (pNode1 != pNode2){
            pNode1 = pNode1->next;
            pNode2 = pNode2->next;
        }
        return pNode1;

    }
};

相似题目

本题与LeetCode中的160. Intersection of Two Linked Lists
一致。参考代码见:
LeetCode 160 code
还可以在牛客网 剑指offer上完成对本题的练习。

面试题45: 圆圈中最后剩下的数字

题目: 0,1,...,n-1这n个数字排成一个圆圈,从数字0开始每次从这个圆圈里删除第m个数字。求出这个圆圈里剩下的最后一个数字。

题目分析

本题是有名的约瑟夫环问题,有两种常见的解法。
第一种是以环形链表模拟圆圈的经典解法,第二种是分析环的规律求解。下面介绍经典的解法。
如下图所示为环形链表模拟圆圈示意图:

环形链表模拟圆圈示意图

参考代码中以std::list来实现环形链表。由于list本身不是一个环形结构,因此每次迭代器扫描到链表末尾的时候,都要将迭代器移到链表的头部来模拟环形链表的遍历。

参考代码

#include<iostream>
#include<list>

using namespace std;

/*===========常规做法:循环链表==============*/
class Solution {
public:
    int LastRemaining_Solution(int n, int m)
    {
        if (n < 1 || m < 1)
            return -1;
        list<int> numbers;
        for (int i = 0; i < n; ++i){
            numbers.push_back(i);
        }

        list<int>::iterator current = numbers.begin();
        while (numbers.size() > 1){
            for (int i = 1; i < m; ++i){    //找到要删除的元素
                current++;
                if (current == numbers.end())   //尾后迭代器不是数组中最后一个元素,而是最后一个元素后面的一个地址
                    current = numbers.begin();
            }

            list<int>::iterator next = ++current;   //list的vector不支持+
            if (next == numbers.end())
                next = numbers.begin();
            --current;
            numbers.erase(current);
            current = next;
        }
        return *current;
    }
};

int main()
{
    Solution solu;
    cout << solu.LastRemaining_Solution(5,3) << endl;

    return 0;
}

相似题目

可以在牛客网 剑指offer上完成对本题的练习。

面试题56: 链表中环的入口结点

题目: 一个链表中包含环,如何找出环的入口结点?

题目分析

本题可以分解为几个子题目解决:
首先,通过快慢指针找到处于链表环中的某个节点。
然后,通过这个节点可以确定链表环的长度。
最后,根据得到环的长度,可以设置快慢指针,找到两个指针第一次相遇的节点即为环的入口节点。

参考代码

/*
struct ListNode {
    int val;
    struct ListNode *next;
    ListNode(int x) :
        val(x), next(NULL) {
    }
};
*/
class Solution {
public:
    ListNode* EntryNodeOfLoop(ListNode* pHead)
    {
        if (pHead == NULL)
            return NULL;
        ListNode* meetingNode = MeetingNode(pHead);
        if (meetingNode == NULL)
            return NULL;

        int length = 1;
        ListNode* pNode = meetingNode->next;
        while (pNode != meetingNode){
            pNode = pNode->next;
            length++;
        }
        //此时length为环的长度

        ListNode* pNode1 = pHead;
        ListNode* pNode2 = pHead;
        for (int i = 0; i < length; ++i){
            pNode2 = pNode2->next;
        }

        while (pNode1 != pNode2){
            pNode1 = pNode1->next;
            pNode2 = pNode2->next;
        }
        return pNode1;
    }
private:
    //通过快慢指针判断是否有环,并且返回环中的一个节点为求环的长度做准备
    ListNode* MeetingNode(ListNode* pHead){
        if (pHead == NULL)
            return NULL;
        ListNode* pSlow = pHead->next;
        if (pSlow == NULL)
            return NULL;
        ListNode* pFirst = pSlow->next;

        while (pSlow != NULL && pFirst != NULL){
            if (pSlow == pFirst)
                return pSlow;
            pSlow = pSlow->next;
            pFirst = pFirst->next;

            if (pFirst != NULL)
                pFirst = pFirst->next;      //快指针每次走两步
        }
        return NULL;    //如果不满足while条件,则证明没有环,返回NULL
    }
};

相似题目

LeetCode中有141. Linked List Cycle为判断链表中是否有环;142. Linked List Cycle II与本题完全一致。这两题的参考代码见:
LeetCode 141 code
LeetCode 142 code

可以在牛客网 剑指offer上完成对本题的练习。

面试题57: 删除链表中重复的结点

题目: 在一个排序的链表中,如何删除重复的结点?

题目分析

本题需要考虑头结点的可删除性。从而需要在头结点前设置哨兵节点,从而遍历链表完成删除操作。

参考代码

/*
struct ListNode {
    int val;
    struct ListNode *next;
    ListNode(int x) :
        val(x), next(NULL) {
    }
};
*/
class Solution {
public:
    ListNode* deleteDuplication(ListNode* pHead)
    {
        if (pHead == NULL)
            return NULL;
        ListNode* pPre = NULL;
        ListNode* pNode = pHead;

        while (pNode != NULL){
            ListNode* pNext = pNode->next;
            bool del = false;
            if (pNext != NULL && pNext->val == pNode->val)
                del = true;

            if (!del){  //不需要删除
                pPre = pNode;
                pNode = pNode->next;
            }
            else{   //需要删除
                int value = pNode->val;
                ListNode* pToBeDel = pNode;
                while (pToBeDel != NULL && pToBeDel->val == value){
                    pNext = pToBeDel->next;
                    delete pToBeDel;
                    pToBeDel = NULL;

                    pToBeDel = pNext;
                }

                if(pPre == NULL)
                    pHead = pNext;
                else
                    pPre->next = pNext;

                pNode = pNext;
            }
        }
        return pHead;
    }
};

class Solution2 {
public:
    ListNode* deleteDuplication(ListNode* pHead)
    {
        ListNode* first = new ListNode(-1); //设置一个起始位置,并初始化为负数
        first->next = pHead;
        ListNode* pNode = pHead;
        ListNode* pre = first;

        while (pNode != NULL && pNode->next != NULL){
            if (pNode->val == pNode->next->val){
                //跳过重复元素
                int val = pNode->val;
                while (pNode != NULL && val == pNode->val){
                    pNode = pNode->next;
                }
                pre->next = pNode;
            }
            else{
                pre = pNode;
                pNode = pNode->next;
            }
        }
        return first->next;
    }
};

相似题目

本题与LeetCode中的82. Remove Duplicates from Sorted List II完全一致,另外LeetCode中还有一道本题的变型,每个重复元素留下一个,即一道链表去重题83. Remove Duplicates from Sorted List
,这两题的参考代码见:
LeetCode 82 code
LeetCode 83 code
还可以在牛客网 剑指offer上完成对本题的练习。

【参考】
[1]《剑指offer》

欢迎转载,转载请注明出处:wenmingxing 《剑指offer》链表专题

上一篇 下一篇

猜你喜欢

热点阅读