链表(Linked List)介绍

链表是有序的列表,但是它在内存中是存储如下:

  • 链表是以节点的方式来存储的,是链式存储。
  • 每个节点包含data域,next域:指向下一个节点。
  • 如图:链表的各个节点不一定是连续存储的。
  • 链表分带头节点的链表和没有头节点的链表,根据实际需求来确定。

单链表(带头结点)逻辑结构示意图:

单链表的应用实例

使用带head头的单向链表实现-水浒英雄排行榜管理

  1. 完成对英雄人物的增删改查操作。
  2. 第一种方法在添加英雄时,直接添加到链表的尾部。
  3. 第二种方法在添加英雄时,根据排名将英雄插入到指定位置(如果有这个排名,则添加失败,并给出提示)

单链的创建示意图

添加(创建)

  1. 先创建一个head头节点,作用就是表示单链表的头。
  2. 后面我们每添加一个节点,就直接加入到链表的最后。
  3. 遍历:通过一个辅助遍历,帮助遍历整个链表。

代码实现(直接添加到链表的尾部)

/**
* @Author Fu~Qiang
* @Time 2021-3-13 19:06:46
* @Version 1.0
* <p>Description:单链表</p>
*/
public class SingleLinkedListDemo { public static void main(String[] args) {
// 测试
HeroNode heroNode1 = new HeroNode(1,"宋江","及时雨");
HeroNode heroNode2 = new HeroNode(2,"卢俊义","玉麒麟");
HeroNode heroNode3 = new HeroNode(3,"吴用","智多星");
HeroNode heroNode4 = new HeroNode(4,"林冲","豹子头");
//创建链表
SingleLinkedList singleLinkedList = new SingleLinkedList();
//加入
singleLinkedList.add(heroNode2);
singleLinkedList.add(heroNode1);
singleLinkedList.add(heroNode3);
singleLinkedList.add(heroNode4);
singleLinkedList.list();
} } //定义一个SingleLinkedList来管理英雄
class SingleLinkedList {
//先初始化一个头节点
private HeroNode head = new HeroNode(0,"","");
//添加节点到单向链表方法
//当不考虑编号的顺序时,找到当前链表最后的节点,将最后这个节点的next指向新的节点
public void add(HeroNode heroNode) {
HeroNode temp = head;
//遍历链表,找到最后
while(true) {
if(temp.next == null) {
break;
}
temp = temp.next;
}
//当退出while循环时,temp就指向了链表的最后
temp.next = heroNode;
}
//显示链表
public void list() {
//判断链表是否为null
if(head.next == null) {
System.out.println("链表为空~~");
return;
}
HeroNode temp = head.next;
while(true) {
//是否到链表最后
if(temp == null) {
break;
}
//输出节点信息
System.out.println(temp);
//将next后移
temp = temp.next;
}
}
} //定义一个heroNode,每个heroNode对象就是一个节点
class HeroNode{
public int no;
public String name;
public String nickName;
public HeroNode next;//指向下一个节点
//构造器
public HeroNode(int no,String name,String nickName) {
this.no = no;
this.name = name;
this.nickName = nickName;
}
//重写toString
@Override
public String toString() {
// TODO Auto-generated method stub
return "HeroNode [no="+no+",name="+name+",nickName="+nickName+"]";
} }

运行截图

按照编号的顺序添加

  1. 首先找到新添加的节点的位置,是通过辅助变量(指针)
  2. 新的节点.next=temp.next
  3. 让temp.next=新的节点

代码实现(按照编号顺序添加)

/**
* @Author Fu~Qiang
* @Time 2021-3-13 19:06:46
* @Version 1.0
* <p>Description:单链表</p>
*/
public class SingleLinkedListDemo { public static void main(String[] args) {
// 测试
HeroNode heroNode1 = new HeroNode(1,"宋江","及时雨");
HeroNode heroNode2 = new HeroNode(2,"卢俊义","玉麒麟");
HeroNode heroNode3 = new HeroNode(3,"吴用","智多星");
HeroNode heroNode4 = new HeroNode(4,"林冲","豹子头");
//创建链表
SingleLinkedList singleLinkedList = new SingleLinkedList();
//加入
singleLinkedList.add(heroNode2);
singleLinkedList.add(heroNode1);
singleLinkedList.add(heroNode3);
singleLinkedList.add(heroNode4);
singleLinkedList.list();
// singleLinkedList.addByOrder(heroNode2);
// singleLinkedList.addByOrder(heroNode1);
// singleLinkedList.addByOrder(heroNode4);
// singleLinkedList.addByOrder(heroNode3);
// singleLinkedList.list();
} } //定义一个SingleLinkedList来管理英雄
class SingleLinkedList {
//先初始化一个头节点
private HeroNode head = new HeroNode(0,"","");
//添加节点到单向链表方法
//当不考虑编号的顺序时,找到当前链表最后的节点,将最后这个节点的next指向新的节点
public void add(HeroNode heroNode) {
HeroNode temp = head;
//遍历链表,找到最后
while(true) {
if(temp.next == null) {
break;
}
temp = temp.next;
}
//当退出while循环时,temp就指向了链表的最后
temp.next = heroNode;
}
//第二种添加英雄的方法
public void addByOrder(HeroNode heroNode) {
HeroNode temp = head;
boolean flag = false;
while(true) {
if(temp.next == null) {//链表最后
break;
}
if(temp.next.no > heroNode.no) {
break;
}else if(temp.next.no == heroNode.no) {//编号已存在
flag = true;
break;
}
temp = temp.next;
}
if(flag) {//flag=true,编号已存在,不能添加
System.out.printf("准备插入的英雄的编号 %d 已经存在",heroNode.no);
}else {
//插入到链表中
heroNode.next = temp.next;
temp.next = heroNode;
} }
//显示链表
public void list() {
//判断链表是否为null
if(head.next == null) {
System.out.println("链表为空~~");
return;
}
HeroNode temp = head.next;
while(true) {
//是否到链表最后
if(temp == null) {
break;
}
//输出节点信息
System.out.println(temp);
//将next后移
temp = temp.next;
}
}
} //定义一个heroNode,每个heroNode对象就是一个节点
class HeroNode{
public int no;
public String name;
public String nickName;
public HeroNode next;//指向下一个节点
//构造器
public HeroNode(int no,String name,String nickName) {
this.no = no;
this.name = name;
this.nickName = nickName;
}
//重写toString
@Override
public String toString() {
// TODO Auto-generated method stub
return "HeroNode [no="+no+",name="+name+",nickName="+nickName+"]";
} }

运行截图

单链表的修改

  1. 先找需要修改的节点,通过遍历。
  2. temp.name = heroNode.name;temp.nickName = heroNode.nickName;

代码实现

//修改节点的信息,根据编号来修改,编号不能修改
public void edit(HeroNode heroNode) {
//判断是否为空
if(head.next == null) {
System.out.println("链表为空~~");
return;
}
HeroNode temp = head.next;
boolean flag = false;
while(true) {
if(temp == null) {
//到链表的最后
break;
}
//找到需要修改的节点
if(temp.no == heroNode.no) {
flag = true;
break;
}
temp = temp.next;
}
if(flag) {
temp.name = heroNode.name;
temp.nickName = heroNode.nickName;
}else{
System.out.printf("没有找到编号%d的节点,不能修改\n",heroNode.no);
}
}

运行截图

单链表的删除

  1. 先找到需要删除的这个节点的前一个节点temp。
  2. temp.next=temp.next.next。
  3. 被删除的节点,将不会有其他引用指向,会被垃圾回收机制回收。

代码实现

//删除节点
public void del(int no) {
HeroNode temp = head;
boolean flag = false;
while(true) {
//已经到链表最后
if(temp.next == null) {
break;
}
//找到待删除节点的前一个节点
if(temp.next.no == no) {
flag = true;
break;
}
temp = temp.next;//temp后移,遍历
}
if(flag) {
temp.next=temp.next.next;
}else{
System.out.printf("要删除的%d不存在,无法删除",no);
}
}

运行截图

【数据结构与算法】——链表(Linked List)的更多相关文章

  1. 数据结构与算法 —— 链表linked list(01)

    链表(维基百科) 链表(Linked list)是一种常见的基础数据结构,是一种线性表,但是并不会按线性的顺序存储数据,而是在每一个节点里存到下一个节点的指针(Pointer).由于不必须按顺序存储, ...

  2. 数据结构与算法 —— 链表linked list(03)

    继续关于linked list的算法题: 删除排序链表中的重复元素 给定一个排序链表,删除所有重复的元素使得每个元素只留下一个. 案例: 给定 1->1->2,返回 1->2 给定  ...

  3. 数据结构与算法 —— 链表linked list(06)

    回文链表 链接 请检查一个链表是否为回文链表. 进阶:你能在 O(n) 的时间和 O(1) 的额外空间中做到吗? 解题思路: 回文链表的特点就是对称. 把链表放到栈中去,利用栈的先进后出的规则,和原链 ...

  4. 数据结构与算法——链表 Linked List(单链表、双向链表、单向环形链表-Josephu 问题)

    链表是有序的列表,但是在内存中存储图下图所示 链表是以 节点 的方式来存储,是 链式存储 每个节点包含 data 域.next 域,指向下一个节点 链表的各个节点 不一定是连续存储,如上图所示 链表还 ...

  5. 数据结构与算法 —— 链表linked list(02)

    我们继续来看链表的第二道题,来自于leetcode: 两数相加 给定两个非空链表来代表两个非负整数,位数按照逆序方式存储,它们的每个节点只存储单个数字.将这两数相加会返回一个新的链表. 你可以假设除了 ...

  6. 数据结构与算法 —— 链表linked list(04)

    我们在上篇文章里面提到了链表的翻转,给定一个链表,对每两个相邻的节点作交换,并返回头节点,今天的这道题是它的升级版,如下: k个一组翻转链表 给出一个链表,每 k 个节点一组进行翻转,并返回翻转后的链 ...

  7. 数据结构与算法 —— 链表linked list(05)

    反转一个单链表. 进阶:链表可以迭代或递归地反转.你能否两个都实现一遍? 示例 : 给定这个链表:1->2->3->4->5 返回结果: 5->4->3->2 ...

  8. JavaScript数据结构与算法-链表练习

    链表的实现 一. 单向链表 // Node类 function Node (element) { this.element = element; this.next = null; } // Link ...

  9. python数据结构与算法——链表

    具体的数据结构可以参考下面的这两篇博客: python 数据结构之单链表的实现: http://www.cnblogs.com/yupeng/p/3413763.html python 数据结构之双向 ...

  10. Java数据结构和算法 - 链表

    Q: 为什么要引入链表的概念?它是解决什么问题的? A: 数组作为数据存储结构有一定的缺陷,在无序数组中,搜索是低效的:而在有序数组中,插入效率又很低:不管在哪一个数组中删除效率都很低:况且一个数组创 ...

随机推荐

  1. String的20个方法

    String的20个方法 面试题 1.new和不new的区别 String A="OK"; String B="OK";//会去常量池查找有没有"Ok ...

  2. Dapr 交通控制示例

    Dapr 已在塔架就位 将发射新一代微服务 牛年 dotnet云原生技术趋势 Dapr是如何简化微服务的开发和部署 前面几篇文章都是从大的方面给大家分享Dapr 能帮助我们解决什么问题,微软从开源到1 ...

  3. GraphQL All In One

    GraphQL All In One refs https://github.com/hasura/learn-graphql xgqfrms 2012-2020 www.cnblogs.com 发布 ...

  4. You Don't Know the Hack tech in the frontend development

    You Don't Know the Hack tech in the frontend development 你所不知道的前端黑科技 css in js animation https://www ...

  5. input number css hidden arrow

    input number css hidden arrow show arrow OK input[type="number"]::-webkit-inner-spin-butto ...

  6. 「NGK每日快讯」2021.1.26日NGK公链第84期官方快讯!

  7. NGK和USDN的应用

    一.NGK和USDN的发展方向 目前区块链将会朝着两个方向去发展,第一种是金融经济的衍生品,第二种是商业应用,快速支付的货币体系,NGK.IO公链是基于分布式应用设计的商用金融区块链操作系统,通过数字 ...

  8. JVM Attach实现原理剖析

    本文转载自JVM Attach实现原理剖析 前言 本文旨在从理论上分析JVM 在 Linux 环境下 Attach 操作的前因后果,以及 JVM 为此而设计并实现的解决方案,通过本文,我希望能够讲述清 ...

  9. 开启算法之路,还原题目,用debug调试搞懂每一道题

    文章简述 大家好,本篇是个人的第 3 篇文章. 承接第一篇文章<手写单链表基础之增,删,查!附赠一道链表题>,在第一篇文章中提过,在刷算法题之前先将基础知识过一遍,这样对后面的做算法题是很 ...

  10. JAVA 批量下载服务器文件到本地指定文件夹并重命名

    /** * @功能 下载文件到指定文件夹并重命名 * @param url 请求的路径 * @param filePath 文件将要保存的目录 * @param filename 保存到本地的文件名 ...