深圳幻海软件技术有限公司 欢迎您!

Java语言----LinkedList 和 链表的实现

2023-05-31

目录一.链表概念二.链表的分类 三.无头单向非循环链表的实现3.1创建简单链表3.2链表基本方法实现3.3四大基本功能        3.3.1.增加元素结点    &n

目录

一.链表概念

二.链表的分类 

三.无头单向非循环链表的实现

3.1创建简单链表

3.2 链表基本方法实现

3.3四大基本功能

             3.3.1.增加元素结点

             3.3.2.查找元素结点

             3.3.3.删除元素结点

             3.3.4.结点信息修改

 

四.LinkedList是什么?

五.LinkedList使用方法

总结


😽个人主页: tq02的博客_CSDN博客-C语言,Java,Java数据结构领域博主
 🌈梦的目标:努力学习,向Java进发,拼搏一切,让自己的未来不会有遗憾。
 🎁欢迎各位→点赞👍 + 收藏⭐ + 评论📝+关注✨
  本章讲解内容:链表 的讲解

 

使用编译器:IDEA 

一.链表概念

         链表是一种物理存储结构上非连续存储结构,数据元素的逻辑顺序是通过链表中的引用链接次序实现的 。

逻辑结构:

注:1、如上图,相当于火车车厢,每一节都相连在一起。

       2、各个结点连接的方式:通过地址连接,在内存当中,相邻结点在内存中不一定相邻。

       3、所有结点都在  堆 中申请出来。

       4、 每个结点包括两个部分:一个是存储数据元素的数据域,另一个是存储下一个结点地址的指针域


二.链表的分类 

 1.单向、双向链表

 注:无论单向还是双向,都是一个结点存储着下(上)一个结点。

2.带头、不带头结点 链表

 注:无头和带头结点的主要区别:有一个起始结点。

3.循环、非循环链表 

 循环链表,就是指:头、尾结点有联系。

       在链表结构中,这是主要的链表,但是这些链表种类还可以结合,如:带头双向循环链表、双向循环链表等等。

链表的种类很多,但都大同小异,我们主要学习两种链表:

    1、无头单向非循环链表:结构简单,一般不会单独用来存数据。实际中更多是作为其他数据结构的子结构,如哈希桶、图的邻接表等等。另外这种结构在笔试面试中出现很多。              2、无头双向链表:在Java的集合框架库中LinkedList底层实现就是无头双向循环链表


三.无头单向非循环链表的实现

3.1创建简单链表

                                  重点:每个结点存储着下一个结点的地址。

创建链表代码实现:

  1. public class SingleLinkedList {
  2. static class List{
  3. int item; //存储数据
  4. List next; //指向下一个结点
  5. public List(int item) {
  6. this.item = item;
  7. }
  8. public List() {};
  9. }
  10. //各种链表实现方法
  11. //头插法
  12. public void addFirst(int data){
  13. }
  14. //尾插法
  15. public void addLast(int data){
  16. }
  17. //任意位置插入,第一个数据节点为0号下标
  18. public void addIndex(int index,int data){
  19. }
  20. //查找是否包含关键字key是否在单链表当中
  21. public boolean contains(int key){
  22. return false;
  23. }
  24. //删除第一次出现关键字为key的节点
  25. public void remove(int key){
  26. }
  27. //得到单链表的长度
  28. public int size(){
  29. return -1;
  30. }
  31. //链表的清空
  32. public void clear() {
  33. }
  34. //展示链表
  35. public void display() {}

3.2 链表基本方法实现

1.遍历链表元素

  1. public void show() {
  2. //这里不是定义了一个节点 这里只是一个引用
  3. ListNode cur = head;
  4. while (cur != null) {
  5. System.out.print(cur.val+" ");
  6. cur = cur.next;
  7. }
  8. System.out.println();
  9. }

2.获取链表长度

  1. public int size(){
  2. int count = 0;
  3. ListNode cur = head;
  4. while (cur != null) {
  5. count++;
  6. cur = cur.next;
  7. }
  8. return count;
  9. }

 3.查询数据

  1. public boolean contains(int key){
  2. ListNode cur = head;
  3. while (cur != null) {
  4. //如果val值 是引用类型 那么这里得用equals来进行比较!!!
  5. if(cur.val == key) {
  6. return true;
  7. }
  8. cur = cur.next;
  9. }
  10. return false;
  11. }

4.链表的清空

  1. public void clear() {
  2. //将所有结点都置空,更为安全
  3. while (head != null) {
  4. ListNode headNext = head.next;
  5. head.next = null;
  6. head = headNext;
  7. }
  8. }


3.3四大基本功能

      3.3.1 、增加元素结点

  1.头插法:将新增结点放在链表的头部。

  1. public void addFirst(int data){
  2. ListNode node = new ListNode(data);
  3. node.next = head;
  4. head = node;
  5. }

2.尾插法:将新增结点直接连接在链表的尾部

  1. public void addLast(int data){
  2. ListNode node = new ListNode(data);
  3. if(head == null) {
  4. head = node;
  5. return;
  6. }
  7. ListNode cur = head;
  8. while (cur.next != null) {
  9. cur = cur.next;
  10. }
  11. //cur 指向的节点就是尾巴节点
  12. cur.next = node;
  13. }

3.选择下标值,添加结点

  1. public void addIndex(int index,int data){
  2. int len = size();
  3. //0、判断index位置的合法性
  4. if(index < 0 || index > len) {
  5. throw new IndexOutOfBounds("任意位置插入数据的时候,index位置不合法: "+index);
  6. }
  7. if(index == 0) {
  8. addFirst(data);
  9. return;
  10. }
  11. if(index == len) {
  12. addLast(data);
  13. return;
  14. }
  15. //1、先找到index-1位置的节点
  16. ListNode cur = findIndex(index);
  17. //2、进行插入
  18. ListNode node = new ListNode(data);
  19. node.next = cur.next;
  20. cur.next = node;
  21. }

 3.3.2.查找元素结点

查找一个元素,返回对应的下标值。

  1. public ListNode findIndex(int index) {
  2. ListNode cur = head;
  3. while (index - 1 != 0) {
  4. cur = cur.next;
  5. index--;
  6. }
  7. return cur;//index-1位置的节点
  8. }

3.3.3.删除元素结点

先找到对应的下标值,然后进行删除。删除方法,前一个结点连接到删除结点的后一个结点。

 如图,先断开d2与d3的连接,然后d2直接连接d4

代码实现:

  1. //删除第一次出现关键字为key的节点
  2. public void remove(int key){
  3. if(head == null) {
  4. return;
  5. }
  6. //当删除结点为头结点
  7. if(head.val == key) {
  8. head = head.next;
  9. return;
  10. }
  11. ListNode prev = searchPrev(key); //返回待删除结点的前一个结点
  12. if(prev == null) {
  13. System.out.println("没有这个数据!");
  14. return;
  15. }
  16. ListNode del = prev.next;
  17. prev.next = del.next;
  18. }
  19. private ListNode searchPrev(int key) {
  20. ListNode prev = head;
  21. while (prev.next != null) {
  22. if(prev.next.val == key) {
  23. return prev;
  24. }else {
  25. prev = prev.next;
  26. }
  27. }
  28. return null;
  29. }

3.3.4.结点信息修改

修改指定下标值的结点元素

  1. public void searchPrev(int num,int date) {
  2. ListNode prev = head;
  3. for(int i=0;i<num-1;i++) {
  4. prev = prev.next;
  5. }
  6. prev.val=date;
  7. }

四.LinkedList是什么?

        LinkedList的底层是双向链表结构,由于链表没有将元素存储在连续的空间中,元素存储在单独的节点中,然后通过引用将节点连接起来了,因此在在任意位置插入或者删除元素时,不需要搬移元素,效率比较高。

 

 如图所示:1. LinkedList实现了List接口
                   2. LinkedList的底层使用了双向链表
                   3. LinkedList没有实现RandomAccess接口,因此LinkedList不支持随机访问。                                 4. LinkedList的任意位置插入和删除元素时效率比较高,时间复杂度为O(1)
                   5. LinkedList比较适合任意位置插入的场景

五.LinkedList使用方法

方法解释
   构造方法LinkedList()   无参构造
public LinkedList(Collection<? extends E> c)使用其他集合容器中元素构造List
常用方法boolean add(E e)尾插e
void add(int index,E element)将e插入到index位置
boolean addAII(Collection<? extends E> c)尾插c中的元素
E remove(int index)删除index位置元素
boolean remove(Object o)删除遇到的第一个o
E get( int index)获取下标index位置元素
void clear()清空

总结

         链表入门很简单,但是却有很多很难的题目,而我们只有将这些题目全部掌握了,对之后的栈、堆等题有更加深刻的知识。

题目推荐:1、移除链表元素        2、反转单链表       3、返回链表中间节点                                 

                 4、 合并两个有序链表       5、 合并两个有序链表       6. 相交链表       7、回文链表

文章知识点与官方知识档案匹配,可进一步学习相关知识
算法技能树首页概览47407 人正在系统学习中