3.5 MyLinkedList 类的实现

MyLinkedList 将用双链表实现,并且还需要保留该表两端的引用。这将需要三个类

  1. MyLinkedList 类,包含到两端的链、表的大小以及一些方法。
  2. Node 类 一个私有的嵌套类。一个节点包含数据以及到前一个节点的链和到下一个节点的链。
  3. LinkedListIterator 类,是一个私有类,并实现接口 Iterator。提供 next、hasNext 和 remove 的实现。

实现中会在链表的头尾增加额外的标记节点,前端的头结点和末端的尾节点。如果不使用头结点,那么对第 1 个结点的处理就会变成特殊情况,比如删除时要访问被删除节点之前的一个节点。

public class MyLinkedList<E> implements Iterable<E> {

    private int theSize;
private int modCount = 0;
private Node<E> beginMarker;
private Node<E> endMarker;
//希望Node类只能被MyLinkedList访问且,Node没有用到外部类的方法和属性,因此设置为静态内部类
private static class Node<E> { public E data;//public 访问属性也不会破坏封装,因为Node在MyLinkedList中是private的
public Node<E> prev;
public Node<E> next; public Node(E data, Node<E> prev, Node<E> next) {
this.data = data;
this.prev = prev;
this.next = next;
}
} private void doClear() {
this.beginMarker = new Node<E>(null, null, null);
this.endMarker = new Node<E>(null, beginMarker, null);
beginMarker.next = endMarker;
theSize = 0;
modCount++;
} public MyLinkedList() {
doClear();
} public void clear() {
doClear();
} public int size() {
return theSize;
} public boolean isEmpty() {
return theSize == 0;
} public boolean add(E e) {
add(theSize, e);
return true;
} public void add(int idx, E e) {
addBefore(getNode(idx, 0, theSize), e);
} public E get(int idx) {
return getNode(idx).data;
} public E set(int idx, E newVal) {
Node<E> p = getNode(idx);
E oldVal = p.data;
p.data = newVal;
return oldVal;
} public E remove(int idx) {
return remove(getNode(idx));
} private void addBefore(Node<E> p, E e) {
Node<E> newNode = new Node<>(e, p.prev, p);
newNode.prev.next = newNode;
p.prev = newNode;
theSize++;
modCount++;
} private E remove(Node<E> p) {
p.next.prev = p.prev;
p.prev.next = p.next;
theSize--;
modCount++;
return p.data;
} private Node<E> getNode(int idx) {
return getNode(idx, 0, theSize - 1);
} private Node<E> getNode(int idx, int lower, int upper) {
Node<E> p;
if (idx < lower || idx > upper) {
throw new IndexOutOfBoundsException();
} if (idx < theSize / 2) {
p = beginMarker.next;
for (int i = 0; i < idx; i++) {
p = p.next;
}
} else {
p = endMarker;
for (int i = theSize; i > idx; i--) {
p = p.prev;
}
}
return p;
} public Iterator<E> iterator() {
return new LinkedListIterator();
} private class LinkedListIterator implements Iterator<E>{ private Node<E> current = beginMarker.next;
private int expectedModCount = modCount;
private boolean okToRemove = false; public boolean hasNext() {
return current != endMarker;
} public E next() {
if (modCount != expectedModCount){//避免在使用迭代器时,被迭代器之外的方法修改了List
throw new ConcurrentModificationException();
}
if(!hasNext()){
throw new NoSuchElementException();
} E nextItem = current.data;
current = current.next;
okToRemove = true;
return nextItem;
} public void remove(){
if (modCount != expectedModCount){
throw new ConcurrentModificationException();
}
if (!okToRemove){
throw new IllegalStateException();
}
MyLinkedList.this.remove(current.prev);
expectedModCount++;
okToRemove = false;
}
}
}

3.5 MyLinkedList 实现的更多相关文章

  1. MyLinkedList

    /** * 节点类 * @author JP * */ class Node { Object value;//节点元素值 Node pre;//上一个节点 Node next;//下一个节点 pub ...

  2. 深入理解java中的ArrayList和LinkedList

    杂谈最基本数据结构--"线性表": 表结构是一种最基本的数据结构,最常见的实现是数组,几乎在每个程序每一种开发语言中都提供了数组这个顺序存储的线性表结构实现. 什么是线性表? 由0 ...

  3. Java实现单链表的各种操作

    Java实现单链表的各种操作 主要内容:1.单链表的基本操作 2.删除重复数据 3.找到倒数第k个元素   4.实现链表的反转   5.从尾到头输出链表 6.找到中间节点 7.检测链表是否有环 8.在 ...

  4. 数据结构(Java描述)之线性表

    基础概念 数据结构:是相互之间存在一种或多种关系的数据元素的集合. 逻辑结构和物理结构 关于数据结构,我们可以从逻辑结构和物理结构这两个维度去描述 逻辑结构是数据对象中数据元素之间的关系,是从逻辑意义 ...

  5. 自定义Java集合

    一.泛型 1.在JDK1.4以前,所有的集合元素全都按照Object来存储,拿出来还要进行强制转型.由于这样的做法有太多的缺点,容易出现ClassCaseException,不安全,让人不省心,于是乎 ...

  6. Java Iterator, ListIterator 和 foreach语句使用

    Java Iterator, ListIterator 和 foreach语句使用 foreach语句结构: for(part1:part2){part3};  part2 中是一个数组对象,或者是带 ...

  7. java基础语法要点<二>(基于1.8)

    注解(元数据) 从jdk5 开始,java支持在源文件中嵌入补充信息,称为注释(annotation).注释不会改变程序的动作,也就不会改变程序的语义.但在开发和部署期间,各种工具可以使用这类信息.元 ...

  8. 约瑟夫环的java解决

    总共3中解决方法,1.数学推导,2.使用ArrayList递归解决,3.使用首位相连的LinkedList解决 import java.util.ArrayList; /** * 约瑟夫环问题 * 需 ...

  9. [AaronYang]C#人爱学不学[4]

    本文章不适合入门,只适合有一定基础的人看.我更相信知识细节见高低,我是从4.0开始学的,终于有时间系统的学习C#5.0,是5.0中的知识,会特殊标记下.但写的内容也可能含有其他版本framework的 ...

随机推荐

  1. 063 01 Android 零基础入门 01 Java基础语法 08 Java方法 01 无参无返回值方法

    063 01 Android 零基础入门 01 Java基础语法 08 Java方法 01 无参无返回值方法 本文知识点:无参无返回值方法 无参无返回值方法 案例 为什么使用方法?--方便复杂问题调用 ...

  2. C++ 双冒号开头的语法是什么

    z转载:https://blog.csdn.net/LHHopencv/article/details/78353380 命名空间限定.std::string 表示std命名空间下的 string类. ...

  3. 洛谷比赛 「EZEC」 Round 4

    洛谷比赛 「EZEC」 Round 4 T1 zrmpaul Loves Array 题目描述 小 Z 有一个下标从 \(1\) 开始并且长度为 \(n\) 的序列,初始时下标为 \(i\) 位置的数 ...

  4. DES加解密算法(C语言实现)

    DES加密和解密算法的实现(C语言) 主要是做个记录,害怕以后代码丢了,先放到这里了. DES再不进行介绍了,可以看上一篇的 DES 的python实现 转载请注明出处:https://www.cnb ...

  5. JVM 第二篇:垃圾收集器以及算法

    本文内容过于硬核,建议有 Java 相关经验人士阅读. 0. 引言 一说到 JVM ,大多数人第一个想到的可能就是 GC ,今天我们就来聊一聊和 GC 关系最大的垃圾收集器以及垃圾收集算法,希望能通过 ...

  6. devops-jenkins部署和基本使用

    1. jenkins部署和基本使用  1.1) 先关闭centos 7的自带防火墙和selinux [root@test-2 ~]# /bin/systemctl stop firewalld [ro ...

  7. Angluar2 项目搭建

    一 使用 Angular CLI 官方脚手架 1.安装 cli npm install -g @angular/cli 2.创建工作空间和初始应用 ng new my-app 二 tsLint 代码格 ...

  8. 如何获取前端提交来得json格式数据

    composer.json { "require": { "guzzlehttp/guzzle": "~6.0" } } composer ...

  9. xpath取其中几个使用position

    from lxml import etree html = ''' <!DOCTYPE html> <html lang="en"> <head> ...

  10. CC2530定时器模模式最大值计算

    首先假设 频率: f 分频系数: n 间隔定时: s 周期: T 模模式最大值: N 因为 T = 1 / f 所以 s = ( n / f ) * N  =  n * N / f 由此可得 计算模模 ...