【数据结构与算法】04 - 双向链表

双向链表,又称为双链表,是链表的一种,它的每个数据结点中都有两个指针,分别指向直接后继和直接前驱。所以,从双向链表中的任意一个结点开始,都可以很方便地访问它的前驱结点和后继结点。

双向链表

在实现双向链表的时候, addremovenode 方法需要重新实现,链表结点需要进行添加一个前驱结点,需要添加一个指向链表尾节点的指针。其他方法可以参考单向链表 【数据结构与算法】03 - 单向循环链表

1. 修改Node实体

在实体中增加一个前驱结点;

    /**
     * 创建一个节点实体
     *
     * @param <E>
     */
    static class Node<E> {
        Node<E> prev;
        E element;
        Node<E> next;

        public Node(Node<E> prev, E element, Node<E> next) {
            this.prev = prev;
            this.element = element;
            this.next = next;
        }

        /**
         * 重写节点的 toString 方法
         *
         * @return
         */
        @Override
        public String toString() {
            StringBuilder str = new StringBuilder();

            if (prev != null) {
                str.append(prev.element);
            } else {
                str.append("null");
            }

            str.append("_").append(element).append("_");

            if (next != null) {
                str.append(next.element);
            } else {
                str.append("null");
            }
            return str.toString();
        }
    }

2. 修改node方法实现

node方法用于获取指定 index位置的结点并返回,为了提高链表的查找效率,使用向前获取向后查找的方式。当需要查找的index位于总结点数的前半段的时候从前往后进行查找,当位于后半段的时候从后往前进行查找:

    /**
     * 获取指定 index 位置处的节点
     *
     * @param index
     * @return
     */
    private Node<E> node(int index) {
        rangeCheckIndex(index);
        // 因为是双向链表可以进行从前查找或者从后查找的原则
        if (index < size >> 1) {
            Node<E> node = first;
            // 如果index是在链表的前半段则进行从前往后查找的方式
            for (int i = 0; i < index; i++) {
                node = node.next;
            }
            return node;
        } else {
            Node<E> node = last;
            for (int i = size - 1; i > index; i--) {
                node = node.prev;
            }
            return node;
        }
    }

3. 对 add添加结点的方法进行实现

/**
     * 向指定 index 位置插入节点
     *
     * @param index
     * @param element
     */
    @Override
    public void add(int index, E element) {
        rangeCheckIndexForAdd(index);
        // 当在最后插入节点或者是 链表中没有节点得时候
        if (index == size) {
            // 只想原始得尾指针
            Node<E> oldLast = last;
            last = new Node<>(oldLast, element, null);
            if (oldLast == null) {
                first = last;
            } else {
                oldLast.next = last;
            }
        } else {
            // 需要找到需要插入节点位置处得原始节点
            // 这是即将插入节点得下一个节点
            Node<E> next = node(index);
            // 即将插入节点得上一个节点
            Node<E> prev = next.prev;
            Node<E> node = new Node<>(prev, element, next);
            next.prev = node;

            if (prev == null) {
                // 表示是在第一个节点前插入得时候
                first = node;
            } else {
                prev.next = node;
            }
        }
        size++;
    }

4. 对 remove进行修改

 /**
     * 删除指定 index 位置的节点
     *
     * @param index
     * @return
     */
    @Override
    public E remove(int index) {
        rangeCheckIndex(index);
        // 获取需要删除的节点
        Node<E> node = node(index);
        Node<E> prev = node.prev;
        Node<E> next = node.next;

        // 可能上一个节点时 null
        if (prev == null) {
            first = next;
        } else {
            prev.next = next;
        }

        if (next == null) {
            last = prev;
        } else {
            next.prev = prev;
        }
        size--;
        return node.element;
    }

5. 对 clear 清空链表的方法进行修改

 /**
     * 清空链表
     */
    @Override
    public void clear() {
        first = null;
        last = null;
        size = 0;
    }

6. 全部实现代码

6.1 接口定义
public interface List<T> {

    final static int ELEMENT_NOT_FOUND = -1;

    int size();

    boolean isEmpty();

    boolean contains(T element);

    void add(T element);

    void add(int index, T element);

    T get(int index);

    T set(int index, T element);

    T remove(int index);

    int indexOf(T element);

    void clear();
}
6.2 抽象类实现公共的方法

/**
 * Description: dsai
 * Created by itLu on 2021/6/26 10:45
 *
 * @author yanpeng.lu
 */
public abstract class AbstractLinkedList<T> implements List<T> {

    /**
     * 链表中元素的个数
     */
    protected int size;

    /**
     * 获取当前链表中元素的个数
     *
     * @return
     */
    @Override
    public int size() {
        return size;
    }

    /**
     * 判断当前链表是否为空
     *
     * @return
     */
    @Override
    public boolean isEmpty() {
        return size == 0;
    }

    /**
     * 判断链表中是否存在某个元素
     *
     * @param element
     * @return
     */
    @Override
    public boolean contains(T element) {
        return indexOf(element) != ELEMENT_NOT_FOUND;
    }

    /**
     * 向链表的末尾添加一个节点
     *
     * @param element
     */
    @Override
    public void add(T element) {
        add(size, element);
    }

    /**
     * 校验 index 是否合法
     *
     * @param index
     */
    protected void rangeCheckIndex(int index) {
        if (index < 0 || index >= size) {
            outOfBound(index);
        }
    }

    /**
     * 统一处理 index 越界
     *
     * @param index
     */
    private void outOfBound(int index) {
        throw new IndexOutOfBoundsException("index : " + index + " size : " + size);
    }

    /**
     * @param index
     * 针对添加节点的index校验
     */
    protected void rangeCheckIndexForAdd(int index) {
        if (index < 0 || index > size) {
            outOfBound(index);
        }
    }
}

6.3 实现自己独有的方法

/**
 * Description: dsai
 * Created by itLu on 2021/6/26 15:35
 *
 * @author yanpeng.lu
 */
public class DoublyLinkedList<E> extends AbstractLinkedList<E> {

    /**
     * 指向头节点的指针 上一个节点 prev 是 null
     */
    private Node<E> first;

    /**
     * 指向尾节点的指针 下一个节点 next 是 null
     */
    private Node<E> last;

    /**
     * 创建一个节点实体
     *
     * @param <E>
     */
    static class Node<E> {
        Node<E> prev;
        E element;
        Node<E> next;

        public Node(Node<E> prev, E element, Node<E> next) {
            this.prev = prev;
            this.element = element;
            this.next = next;
        }

        /**
         * 重写节点的 toString 方法
         *
         * @return
         */
        @Override
        public String toString() {
            StringBuilder str = new StringBuilder();

            if (prev != null) {
                str.append(prev.element);
            } else {
                str.append("null");
            }

            str.append("_").append(element).append("_");

            if (next != null) {
                str.append(next.element);
            } else {
                str.append("null");
            }
            return str.toString();
        }
    }

    /**
     * 向指定 index 位置插入节点
     *
     * @param index
     * @param element
     */
    @Override
    public void add(int index, E element) {
        rangeCheckIndexForAdd(index);
        // 当在最后插入节点或者是 链表中没有节点得时候
        if (index == size) {
            // 只想原始得尾指针
            Node<E> oldLast = last;
            last = new Node<>(oldLast, element, null);
            if (oldLast == null) {
                first = last;
            } else {
                oldLast.next = last;
            }
        } else {
            // 需要找到需要插入节点位置处得原始节点
            // 这是即将插入节点得下一个节点
            Node<E> next = node(index);
            // 即将插入节点得上一个节点
            Node<E> prev = next.prev;
            Node<E> node = new Node<>(prev, element, next);
            next.prev = node;

            if (prev == null) {
                // 表示是在第一个节点前插入得时候
                first = node;
            } else {
                prev.next = node;
            }
        }
        size++;
    }

    /**
     * 获取指定 index 位置得元素值
     *
     * @param index
     * @return
     */
    @Override
    public E get(int index) {
        return node(index).element;
    }

    /**
     * 覆盖指定 index 位置得元素值
     *
     * @param index
     * @param element
     * @return
     */
    @Override
    public E set(int index, E element) {
        Node<E> node = node(index);
        E oldElement = node.element;
        node.element = element;
        return oldElement;
    }

    /**
     * 删除指定 index 位置的节点
     *
     * @param index
     * @return
     */
    @Override
    public E remove(int index) {
        rangeCheckIndex(index);
        // 获取需要删除的节点
        Node<E> node = node(index);
        Node<E> prev = node.prev;
        Node<E> next = node.next;

        // 可能上一个节点时 null
        if (prev == null) {
            first = next;
        } else {
            prev.next = next;
        }

        if (next == null) {
            last = prev;
        } else {
            next.prev = prev;
        }
        size--;
        return node.element;
    }

    /**
     * 获取指定元素在链表中的位置
     *
     * @param element
     * @return
     */
    @Override
    public int indexOf(E element) {
        Node<E> node = first;
        if (element == null) {
            for (int i = 0; i < size; i++) {
                if (node.element == null) {
                    return i;
                }
                node = node.next;
            }
        } else {
            for (int i = 0; i < size; i++) {
                if (element.equals(node.element)) {
                    return i;
                }
                node = node.next;
            }
        }

        return ELEMENT_NOT_FOUND;
    }

    /**
     * 清空链表
     */
    @Override
    public void clear() {
        first = null;
        last = null;
        size = 0;
    }

    /**
     * 获取指定 index 位置处的节点
     *
     * @param index
     * @return
     */
    private Node<E> node(int index) {
        rangeCheckIndex(index);
        // 因为是双向链表可以进行从前查找或者从后查找的原则
        if (index < size >> 1) {
            Node<E> node = first;
            // 如果index是在链表的前半段则进行从前往后查找的方式
            for (int i = 0; i < index; i++) {
                node = node.next;
            }
            return node;
        } else {
            Node<E> node = last;
            for (int i = size - 1; i > index; i--) {
                node = node.prev;
            }
            return node;
        }
    }

    /**
     * 重写toString方法
     *
     * @return
     */
    @Override
    public String toString() {
        StringBuilder str = new StringBuilder();

        str.append("size : ").append(size).append(" [ ");

        Node<E> node = first;

        for (int i = 0; i < size; i++) {
            if (i != 0) {
                str.append(" , ");
            }
            str.append(node);
            node = node.next;
        }

        str.append(" ] ");

        return str.toString();
    }
}

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,012评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,628评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,653评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,485评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,574评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,590评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,596评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,340评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,794评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,102评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,276评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,940评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,583评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,201评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,441评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,173评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,136评论 2 352

推荐阅读更多精彩内容