美文网首页
13 在O(1)时间删除链表结点

13 在O(1)时间删除链表结点

作者: WalkZeRo | 来源:发表于2016-06-26 22:55 被阅读0次

    题目

    给定一个单向链表和一个结点指针,删除该结点

    解析

    删除一个结点有两种方法

    删除结点的两种方法.jpg

    技巧:我们把下一个结点(j)的内容copy to 覆盖需要删除的结点(i),再把下一个结点(j)删除。

    #include <iostream>
    
    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;
        }
    }
    
    // ====================测试代码====================
    void Test(ListNode* pListHead, ListNode* pNode)
    {
        printf("The original list is: \n");
        PrintList(pListHead);
    
        printf("The node to be deleted is: \n");
        PrintListNode(pNode);
    
        DeleteNode(&pListHead, pNode);
        
        printf("The result list is: \n");
        PrintList(pListHead);
    }
    
    // 链表中有多个结点,删除中间的结点
    void Test1()
    {
        ListNode* pNode1 = CreateListNode(1);
        ListNode* pNode2 = CreateListNode(2);
        ListNode* pNode3 = CreateListNode(3);
        ListNode* pNode4 = CreateListNode(4);
        ListNode* pNode5 = CreateListNode(5);
    
        ConnectListNodes(pNode1, pNode2);
        ConnectListNodes(pNode2, pNode3);
        ConnectListNodes(pNode3, pNode4);
        ConnectListNodes(pNode4, pNode5);
    
        Test(pNode1, pNode3);
    
        DestroyList(pNode1);
    }
    
    // 链表中有多个结点,删除尾结点
    void Test2()
    {
        ListNode* pNode1 = CreateListNode(1);
        ListNode* pNode2 = CreateListNode(2);
        ListNode* pNode3 = CreateListNode(3);
        ListNode* pNode4 = CreateListNode(4);
        ListNode* pNode5 = CreateListNode(5);
    
        ConnectListNodes(pNode1, pNode2);
        ConnectListNodes(pNode2, pNode3);
        ConnectListNodes(pNode3, pNode4);
        ConnectListNodes(pNode4, pNode5);
    
        Test(pNode1, pNode5);
    
        DestroyList(pNode1);
    }
    
    // 链表中有多个结点,删除头结点
    void Test3()
    {
        ListNode* pNode1 = CreateListNode(1);
        ListNode* pNode2 = CreateListNode(2);
        ListNode* pNode3 = CreateListNode(3);
        ListNode* pNode4 = CreateListNode(4);
        ListNode* pNode5 = CreateListNode(5);
    
        ConnectListNodes(pNode1, pNode2);
        ConnectListNodes(pNode2, pNode3);
        ConnectListNodes(pNode3, pNode4);
        ConnectListNodes(pNode4, pNode5);
    
        Test(pNode1, pNode1);
    
        DestroyList(pNode1);
    }
    
    // 链表中只有一个结点,删除头结点
    void Test4()
    {
        ListNode* pNode1 = CreateListNode(1);
    
        Test(pNode1, pNode1);
    }
    
    // 链表为空
    void Test5()
    {
        Test(NULL, NULL);
    }
    
    int main(void)
    {
        Test1();
        Test2();
        Test3();
        Test4();
        Test5();
    
        return 0;
    }
    

    结果

    结果.jpg

    注意

    我们上述代码是基于一个假设:要删除的结点确实在链表中。我们需要O(n)的时间才能判断链表中是否包含某一结点。受O(1)时间的限制,我们不得不把确保结点在链表中的工作交给DeleteNode的调用者。

    相关文章

      网友评论

          本文标题:13 在O(1)时间删除链表结点

          本文链接:https://www.haomeiwen.com/subject/sxkzdttx.html