【数据结构与算法】03 - 单向循环链表

由于动态数组有个明显得缺点:可能会造成内存空间的大量浪费(动态数据实现动态扩容)。
能否做到用多少内存就申请多少内存呢?链表就可以做到这一点,链表是一种链式存储的线性表,所有元素的内存地址都不一定是连续的。

1. 接口设计

链表的大部分接口和动态数组是一致的,可以在接口中定义统一需要实现的方法。将链表和动态数组需要公共实现的部分放在抽象类中进行实现。有差异的方法再通过抽象类的子类进行实现;

1.1 设计一个公共的接口 List
List:
/**
 * Description: dsai
 * Created by itLu on 2021/6/19 15:17
 *  定义动态数组 + 链表 + 双向链表 需要 实现的方法
 * @author yanpeng.lu
 */
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();
}
1.2 AbstractLinkedList 抽象类中实现一些公共的方法
AbstractLinkedList:
/**
 * 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);
        }
    }
}
1.3 在抽象类子类中实现自己独有的方法
SingleCircleLinkedList
/**
 * Description: dsai
 * Created by itLu on 2021/6/26 13:26
 * 单向循环链表
 *
 * @author yanpeng.lu
 */
public class SingleCircleLinkedList<E> extends AbstractLinkedList<E> {

    private Node<E> first;

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

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

        /**
         * 重写节点的 toString 方法
         *
         * @return
         */
        @Override
        public String toString() {
            StringBuilder str = new StringBuilder();
            str.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) {
        // 校验当前的index是否合法
        rangeCheckIndexForAdd(index);
        // 在插入第一个元素 或 在 第一个位置插入元素的时候需要改变 最后一个元素的 next 的指向
        if (index == 0) {
            // 新的节点
            Node<E> newFirst = new Node<>(element, first);
            // 需要获取到最后一个节点 这里需要注意的是 在node中需要使用到first的指向
            // 还有 当链表中一个元素都没有的时候 需要进行处理
            Node<E> oldLast = size == 0 ? newFirst : node(size - 1);
            // 实现循环
            oldLast.next = newFirst;
            // 这里需要在以上三步进行完成之后再改变first的指向
            first = newFirst;
        } else {
            // 找到需要插入节点的前一个节点
            Node<E> prev = node(index - 1);
            prev.next = new Node<>(element, prev.next);
        }
        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 = first;
        if (index == 0) {
            if (size == 0) {
                first = null;
            }else {
                Node<E> last = node(size - 1);
                first = node.next;
                last.next = first;
            }
        } else {
            // 删除中间节点都不会影响到最后一个节点的指向
            // 需要获取需要删除节点的前一个节点
            Node<E> prev = node(index - 1);
            node = prev.next;
            prev.next = node.next;
        }
        size--;
        return node.element;
    }

    /**
     * 获取指定元素在链表中的位置
     *
     * @param element
     * @return
     */
    @Override
    public int indexOf(E element) {
        if (element == null) {
            Node<E> node = first;
            for (int i = 0; i < size; i++) {
                if (node.element == null) {
                    return i;
                }
                node = node.next;
            }
        } else {
            Node<E> node = first;
            for (int i = 0; i < size; i++) {
                // 需要注意的 是 这里element需要写在前面 因为 node.element 可能出现 null 的情况 会造成空指针异常
                if (element.equals(node.element)) {
                    return i;
                }
                node = node.next;
            }
        }
        // 上面都不满足则表示没有找到 返回 -1
        return ELEMENT_NOT_FOUND;
    }

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

    /**
     * 获取指定位置的节点
     *
     * @param index
     * @return
     */
    public Node<E> node(int index) {
        rangeCheckIndex(index);
        Node<E> node = first;
        for (int i = 0; i < index; i++) {
            node = node.next;
        }
        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();
    }
}

2. add 方法实现思路

在实现循环链表的过程中,添加方法是核心,需要进行与单链表区别的处理。处理需要在 index == 0 位置,也就是链表的头节点位置,因为这里会改变到 index == size -1位置下一个节点的指向。如果是在中间插入,不会影响到链表循环,所以处理方式和单链表一致。

    /**
     * 在链表中指定 index 位置添加一个节点
     * @param index
     * @param element
     */
    @Override
    public void add(int index, E element) {
        // 校验当前的index是否合法
        rangeCheckIndexForAdd(index);
        // 在插入第一个元素 或 在 第一个位置插入元素的时候需要改变 最后一个元素的 next 的指向
        if (index == 0) {
            // 新的节点
            Node<E> newFirst = new Node<>(element, first);
            // 需要获取到最后一个节点 这里需要注意的是 在node中需要使用到first的指向
            // 还有 当链表中一个元素都没有的时候 需要进行处理
            Node<E> oldLast = size == 0 ? newFirst : node(size - 1);
            // 实现循环
            oldLast.next = newFirst;
            // 这里需要在以上三步进行完成之后再改变first的指向
            first = newFirst;
        } else {
            // 找到需要插入节点的前一个节点
            Node<E> prev = node(index - 1);
            prev.next = new Node<>(element, prev.next);
        }
        size++;
    }

3. remove方法实现思路

在实现循环链表的删除节点操作时,在删除头节点时也会影响到循环链表。所以在 index == 0时需要对删除节点进行特殊处理。还有就是当链表中size == 0的时候需要进行特殊处理:

    /**
     * 移除指定位置处的元素
     *
     * @param index
     * @return
     */
    @Override
    public E remove(int index) {
        // 需要对 index 进行合法性校验
        rangeCheckIndex(index);
        Node<E> node = first;
        if (index == 0) {
            if (size == 0) {
                first = null;
            } else {
                Node<E> last = node(size - 1);
                // 需要处理 index == 0 的情况 当 index = 0 时 可能出现 index - 1 = -1 的情况
                first = first.next;
                last.next = first;
            }
        } else {
            // 首先需要获取到需要删除节点的前一个节点
            Node<E> prev = node(index - 1);
            node = prev.next;
            prev.next = node.next;
        }
        // 对链表中的元素个数减 1 操作
        size--;
        return node.element;
    }
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,088评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,715评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,361评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,099评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 60,987评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,063评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,486评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,175评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,440评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,518评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,305评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,190评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,550评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,880评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,152评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,451评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,637评论 2 335

推荐阅读更多精彩内容