Node

java 中的 LIinkedList 的数据结构是链表,而链表中每一个元素是节点。

我们先定义一下节点:

  1. package com.xzlf.collection;
  2. public class Node {
  3. Node previous; // 上一个节点
  4. Node next; // 下一个节点
  5. Object element; // 元素数据
  6. public Node(Object element) {
  7. super();
  8. this.element = element;
  9. }
  10. public Node(Node previous, Node next, Object element) {
  11. super();
  12. this.previous = previous;
  13. this.next = next;
  14. this.element = element;
  15. }
  16. }

版本一:基础版本

先创建一个类,完成链表的创建、添加元素、然后重写toString() 方法:

  1. package com.xzlf.collection;
  2. /**
  3. * 自定义一个链表
  4. * @author xzlf
  5. *
  6. */
  7. public class MyLinkedList {
  8. private Node first;
  9. private Node last;
  10. private int size;
  11. public void add(Object obj) {
  12. Node node = new Node(obj);
  13. if(first == null) {
  14. first = node;
  15. last = node;
  16. }else {
  17. node.previous = last;
  18. node.next = null;
  19. last.next = node;
  20. last = node;
  21. }
  22. size++;
  23. }
  24. @Override
  25. public String toString() {
  26. StringBuilder sb = new StringBuilder("[");
  27. Node tmp = first;
  28. while(tmp != null) {
  29. sb.append(tmp.element + ",");
  30. tmp = tmp.next;
  31. }
  32. sb.setCharAt(sb.length() - 1, ']');
  33. return sb.toString();
  34. }
  35. public static void main(String[] args) {
  36. MyLinkedList list = new MyLinkedList();
  37. list.add("a");
  38. list.add("b");
  39. list.add("c");
  40. list.add("a");
  41. list.add("b");
  42. list.add("c");
  43. list.add("a");
  44. list.add("b");
  45. list.add("c");
  46. System.out.println(list);
  47. }
  48. }

测试:

版本二:增加get() 方法

  1. package com.xzlf.collection;
  2. /**
  3. * 自定义一个链表
  4. * 增加get方法
  5. * @author xzlf
  6. *
  7. */
  8. public class MyLinkedList2 {
  9. private Node first;
  10. private Node last;
  11. private int size;
  12. public void add(Object obj) {
  13. Node node = new Node(obj);
  14. if(first == null) {
  15. first = node;
  16. last = node;
  17. }else {
  18. node.previous = last;
  19. node.next = null;
  20. last.next = node;
  21. last = node;
  22. }
  23. size++;
  24. }
  25. public Object get(int index) {
  26. Node tmp = null;
  27. // 判断索引是否合法
  28. if(index < 0 || index > size - 1) {
  29. throw new RuntimeException("索引不合法:" + index);
  30. }
  31. /*索引位置为前半部分,从头部开始找*/
  32. if (index <= size >> 1) {
  33. tmp = first;
  34. for (int i = 0; i < index; i++) {
  35. tmp = tmp.next;
  36. }
  37. }else {
  38. /*索引位置为或半部分,从未部开始找*/
  39. tmp = last;
  40. for (int i = size -1; i > index; i--) {
  41. tmp = tmp.previous;
  42. }
  43. }
  44. return tmp.element;
  45. }
  46. @Override
  47. public String toString() {
  48. StringBuilder sb = new StringBuilder("[");
  49. Node tmp = first;
  50. while(tmp != null) {
  51. sb.append(tmp.element + ",");
  52. tmp = tmp.next;
  53. }
  54. sb.setCharAt(sb.length() - 1, ']');
  55. return sb.toString();
  56. }
  57. public static void main(String[] args) {
  58. MyLinkedList2 list = new MyLinkedList2();
  59. list.add("a");
  60. list.add("b");
  61. list.add("c");
  62. list.add("d");
  63. list.add("e");
  64. list.add("f");
  65. System.out.println(list);
  66. System.out.println(list.get(1));
  67. System.out.println(list.get(4));
  68. }
  69. }

测试:

版本三:增加remove() 方法

  1. package com.xzlf.collection;
  2. /**
  3. * 自定义一个链表
  4. * 增加remove
  5. * @author xzlf
  6. *
  7. */
  8. public class MyLinkedList3 {
  9. private Node first;
  10. private Node last;
  11. private int size;
  12. public void add(Object obj) {
  13. Node node = new Node(obj);
  14. if(first == null) {
  15. first = node;
  16. last = node;
  17. }else {
  18. node.previous = last;
  19. node.next = null;
  20. last.next = node;
  21. last = node;
  22. }
  23. size++;
  24. }
  25. public Object get(int index) {
  26. Node tmp = null;
  27. // 判断索引是否合法
  28. if(index < 0 || index > size - 1) {
  29. throw new RuntimeException("索引不合法:" + index);
  30. }
  31. tmp = getNode(index);
  32. return tmp == null ? null : tmp.element;
  33. }
  34. public void remove(int index) {
  35. Node tmp = getNode(index);
  36. Node up = tmp.previous;
  37. Node down = tmp.next;
  38. if (tmp != null) {
  39. if (up != null) {
  40. up.next = down;
  41. }
  42. if (down != null) {
  43. down.previous = up;
  44. }
  45. // 被删元素是第一个时
  46. if(index == 0) {
  47. first = down;
  48. }
  49. // 被删元素是最后一个时
  50. if(index == size - 1) {
  51. last = up;
  52. }
  53. size--;
  54. }
  55. }
  56. public Node getNode(int index) {
  57. Node tmp = null;
  58. /*索引位置为前半部分,从头部开始找*/
  59. if (index <= size >> 1) {
  60. tmp = first;
  61. for (int i = 0; i < index; i++) {
  62. tmp = tmp.next;
  63. }
  64. }else {
  65. /*索引位置为或半部分,从未部开始找*/
  66. tmp = last;
  67. for (int i = size -1; i > index; i--) {
  68. tmp = tmp.previous;
  69. }
  70. }
  71. return tmp;
  72. }
  73. @Override
  74. public String toString() {
  75. StringBuilder sb = new StringBuilder("[");
  76. Node tmp = first;
  77. while(tmp != null) {
  78. sb.append(tmp.element + ",");
  79. tmp = tmp.next;
  80. }
  81. sb.setCharAt(sb.length() - 1, ']');
  82. return sb.toString();
  83. }
  84. public static void main(String[] args) {
  85. MyLinkedList3 list = new MyLinkedList3();
  86. list.add("a");
  87. list.add("b");
  88. list.add("c");
  89. list.add("d");
  90. list.add("e");
  91. list.add("f");
  92. System.out.println(list);
  93. list.remove(2);
  94. System.out.println(list);
  95. list.remove(0);// 删除第一个元素
  96. System.out.println(list);
  97. list.remove(3);// 删除最后一个元素
  98. System.out.println(list);
  99. }
  100. }

测试:

版本四:插入节点

  1. package com.xzlf.collection;
  2. /**
  3. * 自定义一个链表
  4. * 插入节点
  5. * @author xzlf
  6. *
  7. */
  8. public class MyLinkedList4 {
  9. private Node first;
  10. private Node last;
  11. private int size;
  12. public void add(Object obj) {
  13. Node node = new Node(obj);
  14. if(first == null) {
  15. first = node;
  16. last = node;
  17. }else {
  18. node.previous = last;
  19. node.next = null;
  20. last.next = node;
  21. last = node;
  22. }
  23. size++;
  24. }
  25. public void add(int index, Object obj) {
  26. Node tmp = getNode(index);
  27. Node newNode = new Node(obj);
  28. if(tmp != null) {
  29. Node up = tmp.previous;
  30. up.next = newNode;
  31. newNode.previous = up;
  32. newNode.next = tmp;
  33. tmp.previous = newNode;
  34. }
  35. }
  36. public Object get(int index) {
  37. Node tmp = null;
  38. // 判断索引是否合法
  39. if(index < 0 || index > size - 1) {
  40. throw new RuntimeException("索引不合法:" + index);
  41. }
  42. tmp = getNode(index);
  43. return tmp == null ? null : tmp.element;
  44. }
  45. public void remove(int index) {
  46. Node tmp = getNode(index);
  47. Node up = tmp.previous;
  48. Node down = tmp.next;
  49. if (tmp != null) {
  50. if (up != null) {
  51. up.next = down;
  52. }
  53. if (down != null) {
  54. down.previous = up;
  55. }
  56. // 被删元素是第一个时
  57. if(index == 0) {
  58. first = down;
  59. }
  60. // 被删元素是最后一个时
  61. if(index == size - 1) {
  62. last = up;
  63. }
  64. size--;
  65. }
  66. }
  67. public Node getNode(int index) {
  68. Node tmp = null;
  69. /*索引位置为前半部分,从头部开始找*/
  70. if (index <= size >> 1) {
  71. tmp = first;
  72. for (int i = 0; i < index; i++) {
  73. tmp = tmp.next;
  74. }
  75. }else {
  76. /*索引位置为或半部分,从未部开始找*/
  77. tmp = last;
  78. for (int i = size -1; i > index; i--) {
  79. tmp = tmp.previous;
  80. }
  81. }
  82. return tmp;
  83. }
  84. @Override
  85. public String toString() {
  86. StringBuilder sb = new StringBuilder("[");
  87. Node tmp = first;
  88. while(tmp != null) {
  89. sb.append(tmp.element + ",");
  90. tmp = tmp.next;
  91. }
  92. sb.setCharAt(sb.length() - 1, ']');
  93. return sb.toString();
  94. }
  95. public static void main(String[] args) {
  96. MyLinkedList4 list = new MyLinkedList4();
  97. list.add("a");
  98. list.add("b");
  99. list.add("c");
  100. list.add("d");
  101. list.add("e");
  102. list.add("f");
  103. System.out.println(list);
  104. list.add(1, "hello");
  105. System.out.println(list);
  106. }
  107. }

测试:

版本五:增加泛型,小的封装

  1. package com.xzlf.collection;
  2. /**
  3. * 自定义一个链表
  4. * 增加泛型,小的封装
  5. * @author xzlf
  6. *
  7. */
  8. public class MyLinkedList5<E> {
  9. private Node first;
  10. private Node last;
  11. private int size;
  12. public void add(E element) {
  13. Node node = new Node(element);
  14. if(first == null) {
  15. first = node;
  16. last = node;
  17. }else {
  18. node.previous = last;
  19. node.next = null;
  20. last.next = node;
  21. last = node;
  22. }
  23. size++;
  24. }
  25. public void add(int index, E element) {
  26. checkRange(index);
  27. Node tmp = getNode(index);
  28. Node newNode = new Node(element);
  29. if(tmp != null) {
  30. Node up = tmp.previous;
  31. up.next = newNode;
  32. newNode.previous = up;
  33. newNode.next = tmp;
  34. tmp.previous = newNode;
  35. size++;
  36. }
  37. }
  38. private void checkRange(int index) {
  39. if(index < 0 || index > size - 1) {
  40. throw new RuntimeException("索引不合法:" + index);
  41. }
  42. }
  43. public E get(int index) {
  44. Node tmp = null;
  45. // 判断索引是否合法
  46. checkRange(index);
  47. tmp = getNode(index);
  48. return tmp == null ? null : (E) tmp.element;
  49. }
  50. public void remove(int index) {
  51. checkRange(index);
  52. Node tmp = getNode(index);
  53. Node up = tmp.previous;
  54. Node down = tmp.next;
  55. if (tmp != null) {
  56. if (up != null) {
  57. up.next = down;
  58. }
  59. if (down != null) {
  60. down.previous = up;
  61. }
  62. // 被删元素是第一个时
  63. if(index == 0) {
  64. first = down;
  65. }
  66. // 被删元素是最后一个时
  67. if(index == size - 1) {
  68. last = up;
  69. }
  70. size--;
  71. }
  72. }
  73. private Node getNode(int index) {
  74. checkRange(index);
  75. Node tmp = null;
  76. /*索引位置为前半部分,从头部开始找*/
  77. if (index <= size >> 1) {
  78. tmp = first;
  79. for (int i = 0; i < index; i++) {
  80. tmp = tmp.next;
  81. }
  82. }else {
  83. /*索引位置为或半部分,从未部开始找*/
  84. tmp = last;
  85. for (int i = size -1; i > index; i--) {
  86. tmp = tmp.previous;
  87. }
  88. }
  89. return tmp;
  90. }
  91. @Override
  92. public String toString() {
  93. StringBuilder sb = new StringBuilder("[");
  94. Node tmp = first;
  95. while(tmp != null) {
  96. sb.append(tmp.element + ",");
  97. tmp = tmp.next;
  98. }
  99. sb.setCharAt(sb.length() - 1, ']');
  100. return sb.toString();
  101. }
  102. public static void main(String[] args) {
  103. MyLinkedList5<String> list = new MyLinkedList5<>();
  104. list.add("a");
  105. list.add("b");
  106. list.add("c");
  107. System.out.println(list);
  108. list.add(1, "hello");
  109. System.out.println(list);
  110. System.out.println(list.get(1));
  111. }
  112. }

测试:

现在我们在编辑上使用add() 方法后已经提示要插入String类型的数据了:



以上代码测试运行结果:

以上代码可能还有部分细节上的bug,不过作为理解LinkedList数据结构的练习应该够用了。

理解java容器底层原理--手动实现LinkedList的更多相关文章

  1. 理解java容器底层原理--手动实现HashMap

    HashMap结构 HashMap的底层是数组+链表,百度百科找了张图: 先写个链表节点的类 package com.xzlf.collection2; public class Node { int ...

  2. 理解java容器底层原理--手动实现HashSet

    HashSet的底层其实就是HashMap,换句话说HashSet就是简化版的HashMap. 直接上代码: package com.xzlf.collection2; import java.uti ...

  3. 理解java容器底层原理--手动实现ArrayList

    为了照顾初学者,我分几分版本发出来 版本一:基础版本 实现对象创建.元素添加.重新toString() 方法 package com.xzlf.collection; /** * 自定义一个Array ...

  4. (前篇:NIO系列 推荐阅读) Java NIO 底层原理

    出处: Java NIO 底层原理 目录 1.1. Java IO读写原理 1.1.1. 内核缓冲与进程缓冲区 1.1.2. java IO读写的底层流程 1.2. 四种主要的IO模型 1.3. 同步 ...

  5. Java面试底层原理

    面试发现经常有些重复的面试问题,自己也应该学会记录下来,最好自己能做成笔记,在下一次面的时候说得有条不紊,深入具体,面试官想必也很开心.以下是我个人总结,请参考: HashSet底层原理:(问了大几率 ...

  6. 10分钟看懂, Java NIO 底层原理

    目录 写在前面 1.1. Java IO读写原理 1.1.1. 内核缓冲与进程缓冲区 1.1.2. java IO读写的底层流程 1.2. 四种主要的IO模型 1.3. 同步阻塞IO(Blocking ...

  7. java容器HashMap原理

    1.为什么需要HashMap 前面我们说了ArrayList和LinkedList,它们对容器内的对象都能实现增.删.改.查.遍历等操作, 并且对应不同的情况,我们可以选择不同的List,用以提高效率 ...

  8. 魔鬼在细节,理解Java并发底层之AQS实现

    jdk的JUC包(java.util.concurrent)提供大量Java并发工具提供使用,基本由Doug Lea编写,很多地方值得学习和借鉴,是进阶升级必经之路 本文从JUC包中常用的对象锁.并发 ...

  9. Java 容器源码分析之 LinkedList

    概览 同 ArrayList 一样,LinkedList 也是对 List 接口的一种具体实现.不同的是,ArrayList 是基于数组来实现的,而 LinkedList 是基于双向链表实现的.Lin ...

随机推荐

  1. ajax实现图片上传与进度条

    这里使用的是bootstract的一个插件来实现 详情请查看文档中的进度条 https://v3.bootcss.com/components/ 引入必要的文件 <link href=" ...

  2. 【poj 2429】GCD & LCM Inverse (Miller-Rabin素数测试和Pollard_Rho_因数分解)

    本题涉及的算法个人无法完全理解,在此提供两个比较好的参考. 原理 (后来又看了一下,其实这篇文章问题还是有的……有时间再搜集一下资料) 代码实现 #include <algorithm> ...

  3. 记录---java中jsp页面引入jquery路径的问题

    今天在jsp页面中引入jquery的时候因为路径不对总是报404,网上的方法找到几种试了试但是最后结果还是不生效,遂想起原先的项目中有引入外部jquery的例子,所以立马看了看,发现当时的项目中是用$ ...

  4. SpringMVC里 form提交到controller404 解决方法

    把 <form action="/logon"> 改为 <form action="${pageContext.request.contextPath} ...

  5. jQuery学习笔记01

    1.jQuery介绍 1.1什么是jQuery ? jQuery,顾名思义,也就是JavaScript和查询(Query),它就是辅助JavaScript开发的js类库. 1.2 jQuery核心思想 ...

  6. rem布局和使用js rem动态改变字体大小,自适应

    解决rem文字动态改变字体大小: <script> console.log(window.devicePixelRatio); var iScale = 1; iScale = iScal ...

  7. linux系统部署安装过程

    1. 虚拟环境安装 1.新建虚拟机   2.虚拟机设置 2.系统历程 1.进入系统引导界面进行配置 引导项说明:            1.安装centos 系统                    ...

  8. MySQl 和 Redis

    MySQL MySQL 是关系型数据库,开放源码软件,主要使用持久化存储设备(像磁盘)数据存放在磁盘中,功能强大. 因为磁盘访问速度远远慢于内存,所以访问速度慢 Redis 是非关系型,高性能的key ...

  9. MODIS系列之NDVI(MOD13Q1)一:数据下载(一)基于插件

    引言: 写MODIS数据处理这个系列文章的初衷,主要是为了分享本人处理MODIS数据方面的一些经验.鉴于网上对这方面系统性的总结还比较少,我搜集资料时也是走了许多的弯路,因此希望通过此文让初学者能够更 ...

  10. 彻底卸载----LoadRunner

    保证所有LoadRunner的相关进程(包括Controller.VuGen.Analysis和Agent Process)全部关闭: 备份好LoadRunner安装目录下测试脚本,这些脚本一般存放在 ...