18 删除链表的节点/重复节点

Posted guoxinxin

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了18 删除链表的节点/重复节点相关的知识,希望对你有一定的参考价值。

题目描述: 

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

//链表定义
struct ListNode {
    int val;
    struct ListNode *next;
    ListNode(int x) :
        val(x), next(NULL) {
    }
};

 注意:输入提供了要删除节点的指针!!

测试用例: 

 1)功能测试(从有多个节点的链表中删除中间、头、尾节点;从只有一个节点的链表中删除唯一的节点)

 2)特殊输入测试(头指针为nullptr;指向要删除的节点的指针为nullptr

 

解题思路:

1)常规做法: 平均时间复杂度为O(n)   面试时不建议使用   

 遍历到要删除节点的前一个节点pre,将要删除节点的下一个节点赋值给前一个节点。pre->next = pCurrent->next;

2)不访问当前节点的前一个节点:

因为要删除节点的指针知晓,则其下一个节点是可以访问的。将下一个节点的值(val和next)赋值给当前节点,然后删除下一个节点即可

 pCurrent->val = pNext->val;

 pCurrent->next = pNext->next;

delete pNext;

pNext = nullptr;

 

题目描述: 

题目二:删除链表中重复的节点

在一个排序的链表中,存在重复的结点,请删除该链表中重复的结点,重复的结点不保留,返回链表头指针。 例如,链表1->2->3->3->4->4->5 处理后为 1->2->5

注:虽然题目示例给的重复节点的个数都是2,但是题目中并没有明确指出。因此要考虑重复节点为多个的情况。如1->2->3->3->3->4处理后应该是1->2->4

 

测试用例:

1)功能测试(重复节点位于链表的头部、中间、尾部;链表中没有重复的节点)

2)特殊输入测试(指向链表头节点的指针为nullptr;链表中所有节点都是重复的

 

解题思路:

1)由于链表是排序的,只要考虑相邻元素值是否相等,相等即重复。

如果当前节点与下一个节点值相同,他们是重复的节点,需要被删除。为了保证删除之后链表仍然是相连的,我们要把当前节点的前一个节点(因此要定义pPreNode)与后面值比当前节点值大的节点相连。 

/*
struct ListNode {
    int val;
    struct ListNode *next;
    ListNode(int x) :
        val(x), next(NULL) {
    }
};
*/
class Solution {
public:
    ListNode* deleteDuplication(ListNode* pHead)
    {
        //链表为空时
        if(pHead==nullptr)return pHead;
        
        ListNode* pPreNode = nullptr; //先初始为空
        ListNode* pCurrentNode = pHead;//指向第一个节点
        ListNode* pNext = nullptr;
        while(pCurrentNode!=nullptr){ //pCurrentNode->next!=nullptr
           // ListNode* pNext = pCurrentNode->next;  //pNext可能为空  //不要放到循环里面定义,每次循环都会定义
            pNext = pCurrentNode->next;  //pNext可能为空
            if(pNext!=nullptr && pCurrentNode->val==pNext->val){  //pNext不为空时,才可以访问其val值,即pNext->val存在
                //有重复节点
                //删除重复节点
                int value = pCurrentNode->val;
                ListNode* pNodeToDel = pCurrentNode;
                while(pNodeToDel!=nullptr && pNodeToDel->val==value){  //循环,以遍历多个连续的重复值
                    pNext = pNodeToDel->next;   // pNodeToDel会被删除,要记录一下它的下一个节点,否则之后就访问不到了 
                    delete pNodeToDel;
                    pNodeToDel = nullptr;
                    pNodeToDel = pNext; //移动到下一个节点
                }
                //重新连接链表
                if(pPreNode == nullptr){ //说明时头节点
                    pHead = pNext;
                }else{ //非头节点
                    pPreNode->next = pNext; //not pPreNode = pNext; 由于写错,总报段错误!!! 不能把pPreNode直接指向pNext
                }
                pCurrentNode = pNext; //别忘了也要更新 pCurrentNode
            }else{
                //没有重复节点
                pPreNode = pCurrentNode;
                //pCurrentNode = pNext;
                pCurrentNode = pCurrentNode->next;
            }   //无论是否删除节点,每一次都要更新pPreNode与pCurrentNode,pNext会在每次循环初被更新
        }
        
        return pHead;
    }
};  

 注意:

[1] delete指针后一定先将指针置空。因为不确定后面是否会在使用指针(会赋值),如果不置空,使用未定义的指针是十分危险的行为。

[2] 要区分删除的节点是否是头节点:line36和line38。因为连接的操作是不同的

2)为了统一删除节点后,头节点与其他节点的连接关系。创建一个新的指针指向头指针。

/*
struct ListNode {
    int val;
    struct ListNode *next;
    ListNode(int x) :
        val(x), next(NULL) {
    }
};
*/
class Solution {
public:
    ListNode* deleteDuplication(ListNode* pHead)
    {
        
        if(pHead==nullptr || pHead->next==nullptr) return pHead;
           
        ListNode* newpHead =new ListNode(-1);
        newpHead->next = pHead;
           
        ListNode* pPreNode = newpHead; 
        ListNode* pNode = pHead; 
        ListNode* pNext = nullptr; 
           
        while(pNode!=nullptr){
            pNext = pNode->next;
            if(pNext!=nullptr && pNode->val==pNext->val){
                //有重复节点
                ListNode* pToDel = pNode;
                int value = pNode->val;
                while(pToDel!=nullptr && pToDel->val==value){
                    pNext =  pToDel->next;
                    delete  pToDel;
                    pToDel = nullptr;
                    pToDel = pNext;
                }
                pPreNode->next = pNext;
                pNode = pNext; //勿忘 
            }else{
                //没有重复节点
                pPreNode = pNode;
                pNode = pNext;
            }
        }
        return newpHead->next;

    }
};

3)使用递归方法:

当头节点重复时,直接移动头节点到第一个不重复的节点。头节点不重复时,移动到下一个节点,剩余部分递归调用。这样链表可以只处理头节点重复的情况。

/*
struct ListNode {
    int val;
    struct ListNode *next;
    ListNode(int x) :
        val(x), next(NULL) {
    }
};
*/
class Solution {
public:
    ListNode* deleteDuplication(ListNode* pHead)
    {
        //递归的终止条件
        if(pHead==nullptr || pHead->next==nullptr) return pHead;
        

        ListNode* pNode = nullptr; 
        
        if(pHead->val==pHead->next->val){
            //链表的头节点有重复
            int value = pHead->val;
            
            while(pHead!=nullptr && pHead->val==value){
                pNode = pHead->next;
                delete pHead;
                pHead=nullptr;
                pHead=pNode;
            }
            
            return deleteDuplication(pHead);  //因为输入的是头节点,没有next指针接收,因此直接return
            
        }else{
            //链表不重复
            pHead->next = deleteDuplication(pHead->next); //返回的链表应该连接在头指针的下面
        }
        return pHead;

    }
};

  

  

 

以上是关于18 删除链表的节点/重复节点的主要内容,如果未能解决你的问题,请参考以下文章

剑指Offer-代码的完整性面试题18.2:删除链表中的重复节点

剑指Offer:删除链表的节点18

剑指offer18.删除链表中重复节点

剑指offer--18删除链表的节点

0ms优化剑指 Offer 18. 删除链表的节点

算法剑指 Offer 18. 删除链表的节点