【问题标题】:Insertion Sort algorithm java doubly linked list插入排序算法java双向链表
【发布时间】:2015-11-12 23:35:43
【问题描述】:

我有一个带有哨兵节点的双向链表,我需要使用 O(n^2) 复杂度的插入排序对其进行排序。我已经编写了这段代码,但它没有按预期工作。

对插入排序有什么帮助,特别是对代码有帮助吗?

    public void insertionSort()
    {
        int key,i;
        int size = getSize();
        this.restart();
        for(i = 2; i < size; i++)
        {
            this.next();  // Go to next node
            key = this.getVar(); // get the integer a node holds
            while(this.hasNext() == 1 && position.prev.var < key && position.var != -1)
            {
                position.prev.setNext(position.next);
                position.next.setPrev(position.prev);
                position.setNext(position.prev);
                position.setPrev(position.prev.prev);
                position.prev.setNext(position);
                position.next.setPrev(position);    
                this.goBack(); // go to the previous node
            }                       
        }

    }

大小是我的列表有多少元素。我的哨兵有 var = -1 所以我想在我领先时停下来,这就是我使用它的原因。

position.var != -1

并且this.hasNext() == 1 是真的,只要我们在一个位置 != sentinel node 。

在具体示例中,我的列表中有 35 个整数:

5 9 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 2 1 1 1 1

我想这样对它们进行排序:

9 5 2 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1

更新: 我在插入排序中使用的代码如下:

public int getSize()
        {
            int size = 0;
            this.restart();
            while(this.hasNext() == 1)
            {
                size++;
                this.next();
            }
            return size;
        }

public void restart()
        {
            position = header;
            previous = Sentinel;
        }

public void next()
        {
            previous = position;
            position = position.next;
        }

public int getVar()
        {
            return position.var;
        }

public void goBack()
        {
            position = previous;
            previous = previous.prev;
        }

    public int hasNext()
        {
            if(position.var != -1)
                return 1;
            else 
                return 0;
        }

    public void setNext(Node next)
        {
            this.next = next;
        }

public void setPrev(Node prev) { this.prev = prev; }

另外,我使用列表迭代器。

【问题讨论】:

  • 澄清一下,第一个数列是你排序后的?
  • 不,当我将它们添加到列表时,它是未排序的。在完成添加后,我需要使用插入排序对列表进行排序。
  • 排序后的输出是什么?
  • 您需要发布更多资源以供他人帮助。 goBack()、getRequest() 和 hasNext() 究竟是做什么的?
  • 顺便说一句,你永远不会更新 position(或定义它),所以 position.var != -1 总是相同的值,表明它要么根本不排序,要么永远不会终止(或者可能在之间)。

标签: java sorting insert linked-list insertion-sort


【解决方案1】:

这是我对内循环的分析笔记,你是对的,那里肯定有问题。

position.unlink():越界,邻居变成直接邻居

position.prev.next = position.next;  // TODO 1: position.next.prev = position.prev
position.next.prev = position.prev;  // TODO 2: position.prev.next = position.next
                  ^  Breaks TODO 1, but we can: position.prev.next.prev = position.prev

所以我们仍然需要 TODO 1 和 2,按照这个顺序:

     -- TODO 1: position.prev.next.prev = position.prev
     -- TODO 2: position.prev.next      = position.next

position.insertBefore(position.prev): 将行中更靠后的位置重新排入队列

position.next = position.prev;          // TODO 3: position.prev.prev = position
position.prev = a = position.prev.prev; // TODO 4: a.next = position
//           ^                          //   same: position.prev.next = position   
//           |                          // *Error 1!*
//           + breaks TODO's 1 and 2, *Error 2!*
//           + breaks TODO 3, but we can instead: position.next.prev = position

关于错误1,TODO 3和TODO 4都涉及position.prev,将nextprev都设置为position;这有效地包围了position.prevposition。无论它前进还是后退,它的直接邻居都是position。一步之后,一切都是一样的。有趣的结构——就像你家的前门和后门都进入同一个地方。

错误 2 导致无法更新 position.prev,这是 TODO12 所需要的em> 和 3。我们仍然可以通过使用通过position.prevnext 字段访问a.next 的替代表达式来满足TODO 3,但是TODO 的12 em> 不能再满足了。

position.insertBefore(position.prev),继续:

position.prev.next = position;        // DONE: 4 
position.next.prev = position;        // DONE: 3 

这两行完成了 TODO 3 和 4,所以剩下的就是:

     -- TODO 1: position.prev.next.prev = position.prev
     -- TODO 2: position.prev.next      = position.next
     -- TODO 5: fix Error 1
     -- TODO 6: fix Error 2

修复错误 1 ​​包括确保 TODO 1 和 2 在 TODO 4 之前完成,修复错误 2 确保 TODO 3 在分配 a 之前完成。

你最终完成了 8 个任务;事后看来,对于双向链表和涉及 4 个节点的移动,这并不奇怪。

【讨论】:

    【解决方案2】:

    这应该可以解决问题:

            int j;
            // ...
            for(i = 1; i < size; i++)
            {
                this.restart();          // start at ith node
                for(j = 0; j < i; j++)
                    this.next();
                key = this.getVar();     // same code as before
    

    或为每个外循环使用另一个每次前进一个节点的变量。

    另外,this.hasNext()不应该重命名为this.hasPrev()吗?

    代码的主要部分似乎正确,示例图:

                    // goal: swap B and C
                    // initial state
              p        
        -> -> -> ->
        A  B  C  D
       <- <- <- <-
                    // remove C from list
                    position.prev.setNext(position.next);
                    position.next.setPrev(position.prev);
              p
           ----->
        ->    -> ->
        A  B  C  D
       <- <- <-
             <----
                    // update C next and previous                    
                    position.setNext(position.prev);
                    position.setPrev(position.prev.prev);
           p
        ----->
           -> -> ->
        A  C  B  D
       <- <-    <-
          <----
                    // insert C into list
                    position.prev.setNext(position);
                    position.next.setPrev(position);
           p
        -> -> -> ->
        A  C  B  D
       <- <- <- <-
    

    【讨论】:

      猜你喜欢
      • 2018-11-04
      • 1970-01-01
      • 2023-04-07
      • 1970-01-01
      • 2018-12-25
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多