实现双向链表(带傀儡节点)

Posted Ischanged

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了实现双向链表(带傀儡节点)相关的知识,希望对你有一定的参考价值。

引言

在之前的博文中,我简单的向大家分享了一些链表相关的知识及一些面试题,如果感兴趣的老铁可以去瞧瞧,今天做题遇到要实现带傀儡节点的双向链表,做了下,之前的单向链表中我们也遇到要设置傀儡节点(哨兵节点的题),今天我们就来看一下实现双向链表(带傀儡节点)。

基本思路

**对于链表没说带傀儡节点或者虚拟节点,这个链表没有真正的头结点,但是我们把第一个节点叫做头结点,它起到标识的作用,标识这个链表的头结点
这个头结点的位置随时可能发生这变化,是不固定的,之后通过这个头结点我们要完成一些链表的增删查改。如果带傀儡节点这个节点的位置是固定的,那么以后我们操作链表就以这个傀儡节点作为头结点,完成我们的一些基本的增删查改操作。**这就很简单了,具体思路见我之前双链表的博文。
代码如下内有关键注释:
DoubleLinkedList .java

///实现双向链表(带傀儡节点)代码
  class ListNode {

        public int data;
        public ListNode prev;
        public ListNode next;

        public ListNode(int data) {

            this.data = data;
        }
    }
    public class DoubleLinkedList {
        public ListNode dummyHead;//虚拟头结点
        public ListNode last;//尾结点
        public DoubleLinkedList() {
            this.dummyHead = new ListNode(-1);//傀儡节点
        }

        //找到某一位置的节点,用于删除节点
        public ListNode findIndex(int index) {
            ListNode cur = dummyHead.next;
            while (index != 0) {
                cur = cur.next;
                index--;

            }
            return cur;
        }


        //头插法
        public void addFirst(int data) {
            ListNode node=new ListNode(data);
            //第一次插入一个节点也没有时的情况
            if(this.dummyHead.next==null) {
                node.prev=this.dummyHead;
                this.dummyHead.next=node;
                this.last=node;
                return;
               // this.head=node;//带头了就不用再定义头了
            }else {
                //新插入的节点和他前后的节点连接
                node.next=this.dummyHead.next;
                node.prev=this.dummyHead;
                this.dummyHead.next.prev=node;
                this.dummyHead.next=node;
            }
        }

        //尾插法
        public void addLast(int data) {
            ListNode node=new ListNode(data);
            if(this.dummyHead.next==null&&this.last==null) {
                node.prev=this.dummyHead;
                this.dummyHead.next=node;
                this.last=node;
                return;
            }else {
                this.last.next=node;
                node.prev=this.last;
                this.last=node;
            }
        }

        //任意位置插入,第一个数据节点为0号下标
        public void addIndex(int index, int data) {
            //判断位置的合法性,任意位置插入,第一个数据节点为0号下标
            if (index < 0 || index > size()) {
                System.out.println("输入的位置不合法");
                return;
            } else {
                if (index == 0) {
                    addFirst(data);
                    return;
                }
                if (index == size()) {
                    addLast(data);
                }else {
                    //中间插
                    ListNode node = new ListNode(data);
                    ListNode cur;
                    cur = findIndex(index);
                    cur.prev.next = node;//当index==size时,如果不采用尾插法,会出现空指针异常
                    node.prev = cur.prev;
                    cur.prev = node;
                    node.next = cur;
                }
            }

        }

        //查找是否包含关键字key是否在单链表当中
        public boolean contains(int key) {
            ListNode cur = this.dummyHead.next;
            while (cur != null) {
                if (cur.data == key) return true;
                 cur=cur.next;
            }
            //return false;
            return false;
        }

        //删除第一次出现关键字为key的节点
        public void remove(int key) {
            ListNode cur = this.dummyHead.next;

            while(cur != null) {
                if(cur.data == key) {
                    if(cur == this.dummyHead.next) {
                        //头结点是唯一的一个节点或不是唯一的一个节点的情况是一样的,
                        // 都是下面的代码和之前的不带头结点的双向链表不同
                        this.dummyHead.next = this.dummyHead.next.next; //两步代码是前后承接的
                        this.dummyHead.next.prev = this.dummyHead;
                    } else {
                        cur.prev.next = cur.next;
                        //判断是不是最后一个节点
                        //要删除的节点不是最后一个节点
                        if (cur.next != null) {
                            cur.next.prev = cur.prev;
                        } else {
                            //要删除的节点是最后一个节点
                            this.last = cur.prev;
                        }
                    }
                    return;
                } else {
                    cur = cur.next;
                }
            }
        }

        //删除所有值为key的节点
        public void removeAllKey(int key) {
            ListNode cur = dummyHead.next;
            while (cur != null) {
                //
                if (cur.data == key) {
                    //判断是不是头结点
                    if (cur == this.dummyHead.next) {//注意这里不要出错
                        this.dummyHead.next=this.dummyHead.next.next;
                        this.dummyHead.next.prev=null;

                    } else {
                        cur.prev.next = cur.next;
                        //判断是不是最后一个节点
                        if (cur.next == null) {
                            this.last = cur.prev;
                        } else {
                            cur.next.prev = cur.prev;
                        }


                    }

                }
                cur = cur.next;继续往后走  不要停 直到为null 的时候
            }
        }

        //得到单链表的长度
        public int size() {
            ListNode cur =this.dummyHead.next;
            int count = 0;
            while (cur != null) {
                count++;
                cur = cur.next;
            }
            return count;
        }

        public void display() {

            ListNode cur = this.dummyHead.next;
            while (cur != null) {
                System.out.print(cur.data + " ");
                cur = cur.next;

            }
            System.out.println();

        }

        public void clear() {
            //把所有的节点都置为null
            ListNode cur =  this.dummyHead.next;
            ListNode curNext;
            while (cur != null) {
                curNext = cur.next;
                cur.prev = null;
                cur.next = null;
                cur = curNext;
            }
            this.dummyHead = null;
            this.last = null;


        }
    }


测试类

ublic class TestDemo {
    public static void main(String[] args) {
        DoubleLinkedList doubleLinkedList =new DoubleLinkedList();
        doubleLinkedList.addFirst(1);
        doubleLinkedList.addFirst(2);
        doubleLinkedList.addFirst(3);
        doubleLinkedList.addFirst(0);
        System.out.println(doubleLinkedList.size());
        doubleLinkedList.display();
        System.out.println(doubleLinkedList.contains(0));
        System.out.println("======================================");
        doubleLinkedList.removeAllKey(1);
        doubleLinkedList.display();
        doubleLinkedList.addFirst(0);
        doubleLinkedList.display();
    }
}

🛣️过🉐小🧑🏽‍🤝‍🧑🏼如果9️⃣🉐🉑以🉐话记🉐点👍🏻🍹持👇🏻,🦀🦀
往期文章:
双向链表的实现(双向链表与单向链表的简单区别联系和实现)
链表的概念和结构及基本功能函数的实现(单链表的实现)

以上是关于实现双向链表(带傀儡节点)的主要内容,如果未能解决你的问题,请参考以下文章

实现双向链表(带傀儡节点)

Java数据结构—— 双向链表及带傀儡节点双向链表的实现

翻转链表 (无傀儡节点,递归+迭代)

翻转链表 (无傀儡节点,递归+迭代)

移除链表元素(无傀儡节点,递归+迭代)

移除链表元素(无傀儡节点,递归+迭代)