zoukankan      html  css  js  c++  java
  • 18 删除链表的节点/重复节点(第3章 高质量的代码-代码的完整性)

    题目描述: 

    题目一:在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;
    
        }
    };
    

      

      

  • 相关阅读:
    Visual Studio 2019 使用.Net Core 3.0 一
    Asp.Net真分页技术
    Vue-员工管理系统
    Activex在没有电子秤api的情况下获取串口数据
    C#调用Activex中串口电子秤的数据,并将电子秤的数据显示到前端页面
    C# Datetime.Ticks
    Asp.Net进阶/管家模式+发布订阅模式:练习
    委托解耦
    Asp.Net进阶/值类型与引用类型:复习
    C# 简单日志帮助类LogHelper
  • 原文地址:https://www.cnblogs.com/GuoXinxin/p/10421144.html
Copyright © 2011-2022 走看看