单链表(Java--尚硅谷)
基础知识
大体结构和C++的链表差不多
补充之前不知道的:链表分两类,带和不带头结点的链表 现在才知道,Java没有像C/C++那样的指针
首先创建一个LinkList
类,然后把链表的各个功能添加进去
创建结点
//当不考虑编号顺序时,找到当前链表最后节点,将最后节点next指向新节点
public void add(PersonNode node) {
//因为头节点不能动,先需要一个辅助遍历节点temp
PersonNode temp = head;
while (true) {//遍历链表,找到最后
if (temp.next == null) {
break;
}
temp = temp.next;//如果没到最后,就继续找下去
}//当退出while循环时,temp指向链表的最后
temp.next = node;
}
添加结点(乱序)
public void addByOrder(PersonNode node) {
//因为头节点不移动,仍通过一个辅助变量来找添加的位置
//因为是单链表,所以找的temp时位于添加位置的前一个结点,否则不能插入
PersonNode temp = head;
boolean flag = false;//标识添加的编号是否存在
while (true) {
if (temp.next == null) //说明temp在链表最后
break;
if (temp.next.no > node.no) {//位置找到了,就在temp后面
break;
} else if (temp.next.no == node.no) {//希望添加的编号已存在
flag = true;
break;
}
temp = temp.next;
}
if (flag) {//编号已存在,不能添加
System.out.println("already exist no." + node.no + ", you can't add it.=====");
} else {
node.next = temp.next;
temp.next = node;
}
}//最终还是按照序号来排列
编辑结点
public void edit(PersonNode node) {//这里直接引入外部结点,根据node.no来寻找需要修改的结点
if (head.next == null) {
System.out.println("empty LinkList====");
return;
}
PersonNode temp;
temp = head.next;
boolean flag = false;//判断是否找到该结点
while (true) {
if (temp == null)
break;//到了链表最后
if (temp.no == node.no) {
flag = true;//找到
break;
}
temp = temp.next;
}
//根据flag判断是否找到
if (flag) {
temp.name = node.name;
temp.score = node.score;
} else {
System.out.println("not found no." + node.no + "====");
}
System.out.println("=======");
}
删除结点
public void delete(int no) {
if (head.next == null) {
System.out.println("empty LinkList===");
return;
}
PersonNode temp = head;//temp指向待删除结点的前一个结点
boolean flag = false;
while (true) {
if (temp.next == null) {
break;//已经到链表最后
}
if (temp.next.no == no) {
flag = true;//找到了待删除结点的前一个结点temp
break;
}
temp = temp.next;
}
if (flag) {
temp.next = temp.next.next;
//Java会自动回收未被访问的数据
} else {
System.out.println("not found====");
}
System.out.println("====");
}
打印链表
public void list() {
//判断链表是否为空
if (head.next == null) {
System.out.println("empty LinkList=====");
return;
}
//头节点不动,需要一个辅助变量来遍历
PersonNode temp = head.next;
while (true) {
//判断是否到链表最后
if (temp == null) {
break;
}
//输出节点信息
System.out.println(temp);
//由于之前重写了toString,这里直接打印temp就可以打印出节点的所有信息
temp = temp.next;
}
}
主函数
public static void main(String[] args) {
PersonNode person1 = new PersonNode(1, "Jordan", 56);
PersonNode person2 = new PersonNode(2, "Kobe", 81);
PersonNode person3 = new PersonNode(3, "James", 61);
PersonNode person4 = new PersonNode(4, "Melo", 60);
LinkList list = new LinkList();
//按照顺序添加
// System.out.println("====add====");
// list.add(person1);
// list.add(person2);
// list.add(person3);
// list.add(person4);
//不按照顺序添加
System.out.println("====add by order====");
list.addByOrder(person1);
list.addByOrder(person4);
list.addByOrder(person3);
list.addByOrder(person2);
list.list();
// list.addByOrder(person2);
System.out.println("====edit====");
PersonNode newNode = new PersonNode(3, "Ivring", 57);
list.edit(newNode);
list.list();
//删除结点
System.out.println("====delete====");
list.delete(4);
list.list();
}
完整代码
//SingleLinkList.java
public class SingleLinkList {
public static void main(String[] args) {
PersonNode person1 = new PersonNode(1, "Jordan", 56);
PersonNode person2 = new PersonNode(2, "Kobe", 81);
PersonNode person3 = new PersonNode(3, "James", 61);
PersonNode person4 = new PersonNode(4, "Melo", 60);
LinkList list = new LinkList();
//按照顺序添加
// System.out.println("====add====");
// list.add(person1);
// list.add(person2);
// list.add(person3);
// list.add(person4);
//不按照顺序添加
System.out.println("====add by order====");
list.addByOrder(person1);
list.addByOrder(person4);
list.addByOrder(person3);
list.addByOrder(person2);
list.list();
// list.addByOrder(person2);
System.out.println("====edit====");
PersonNode newNode = new PersonNode(3, "Ivring", 57);
list.edit(newNode);
list.list();
//删除结点
System.out.println("====delete====");
list.delete(4);
list.list();
}
}
//定义一个成员点
class PersonNode {
String name;
public int no;
public int score;
public PersonNode next;//指向下一个结点
public PersonNode(int no, String name, int score) {
this.no = no;
this.name = name;
this.score = score;
}
//为了显示方便,重写toString
@Override
public String toString() {
return "[no=" + no + ",name=" + name + ",score=" + score + "]";
}
}
//定义LinkList管理person
class LinkList {
//初始化头节点,头节点不能动
private PersonNode head = new PersonNode(0, "", 0);
public PersonNode getHead(){
return head;//返回私有成员head
}
//当不考虑编号顺序时,找到当前链表最后节点,将最后节点next指向新节点
public void add(PersonNode node) {
//因为头节点不能动,先需要一个辅助遍历节点temp
PersonNode temp = head;
while (true) {//遍历链表,找到最后
if (temp.next == null) {
break;
}
temp = temp.next;//如果没到最后,就继续找下去
}//当退出while循环时,temp指向链表的最后
temp.next = node;
}
public void addByOrder(PersonNode node) {
//因为头节点不移动,仍通过一个辅助变量来找添加的位置
//因为是单链表,所以找的temp时位于添加位置的前一个结点,否则不能插入
PersonNode temp = head;
boolean flag = false;//标识添加的编号是否存在
while (true) {
if (temp.next == null) //说明temp在链表最后
break;
if (temp.next.no > node.no) {//位置找到了,就在temp后面
break;
} else if (temp.next.no == node.no) {//希望添加的编号已存在
flag = true;
break;
}
temp = temp.next;
}
if (flag) {//编号已存在,不能添加
System.out.println("already exist no." + node.no + ", you can't add it.=====");
} else {
node.next = temp.next;
temp.next = node;
}
}
public void edit(PersonNode node) {
if (head.next == null) {
System.out.println("empty LinkList====");
return;
}
PersonNode temp;
temp = head.next;
boolean flag = false;//判断是否找到该结点
while (true) {
if (temp == null)
break;//到了链表最后
if (temp.no == node.no) {
flag = true;//找到
break;
}
temp = temp.next;
}
//根据flag判断是否找到
if (flag) {
temp.name = node.name;
temp.score = node.score;
} else {
System.out.println("not found no." + node.no + "====");
}
System.out.println("=======");
}
public void delete(int no) {
if (head.next == null) {
System.out.println("empty LinkList===");
return;
}
PersonNode temp = head;//temp指向待删除结点的前一个结点
boolean flag = false;
while (true) {
if (temp.next == null) {
break;//已经到链表最后
}
if (temp.next.no == no) {
flag = true;//找到了待删除结点的前一个结点temp
break;
}
temp = temp.next;
}
if (flag) {
temp.next = temp.next.next;
} else {
System.out.println("not found====");
}
System.out.println("====");
//Java会自动回收未被访问的数据
}
//显示链表(遍历)
public void list() {
//判断链表是否为空
if (head.next == null) {
System.out.println("empty LinkList=====");
return;
}
//头节点不动,需要一个辅助变量来遍历
PersonNode temp = head.next;
while (true) {
//判断是否到链表最后
if (temp == null) {
break;
}
//输出节点信息
System.out.println(temp);
//由于之前重写了toString,这里直接打印temp就可以打印出节点的所有信息
temp = temp.next;
}
}
}
面试题
求单链表中有效结点个数
//如果带头节点的链表,不统计头节点
public static int getLength(PersonNode head) {
if (head.next == null)
return 0;
int length = 0;
PersonNode current = head.next;//定义辅助变量
while (current != null) {
length++;
current = current.next;
}
return length;
}
在主函数中测试
System.out.println(getLength(list.getHead()));
查找单链表中倒数第n个结点(新浪)
思路
- 编写一个方法,接收head结点,同时接收一个index
- index表示是倒数第index个结点
- 先把链表从头到尾遍历,得到链表的总长度 getLength
- 得到size后,我们从链表第一个开始遍历
(size-index)
public static PersonNode findLastIndexNode(PersonNode head, int index){
if (head.next == null)
return null;
int size = getlength(head);//第一次遍历,获取链表大小
if (index <= 0 || index > size)//判断倒数的序号是否超出容量
return null;
PersonNode current = head.next;
for (int i = 0; i < size - index; i++) {
current = current.next;//遍历size-index次
}
return current;
}
测试
PersonNode res = findLastIndexNode(list.getHead(), 1);
System.out.println(res);
单链表反转(腾讯)
思路
- 先定义一个结点
reverseHead = new PersonNode();
- 从头到尾遍历原来的链表,每遍历一个链表,就将其取出,放在新链表最前端
- 原来链表的
head.next = reverseHead.next;
public static void reverseList(PersonNode head) {
if (head.next == null || head.next.next == null)
return;
// 定义一个辅助遍历,帮助遍历原来的链表
PersonNode current = head.next;
PersonNode next = null;//指向当前结点[current]的下一个结点
PersonNode reverseHead = new PersonNode(0, "", 0);
while (current != null) {
next = current.next;//先暂时保存当前结点的下一个结点,后面会用到
current.next = reverseHead.next;//把current的下一个结点指向新链表最前端
reverseHead.next = current;//将current连接到新链表
current = next;//让current后移
/*每一次循环,reverseHead.next都会按照原链表的顺序定位到current,遍历结束,刚到reverseHead.next定位到链表最后一个,
此时把reverseHead.next地址赋给head.next,这样就可以倒着来遍历链表*/
}
head.next = reverseHead.next;//头结点拼接,实现反转
}
测试
reverseList(list.getHead());
list.list();
从尾到头打印单链表(百度)
思路
- 1.先反转后再打印(有个问题:只要求逆序打印,不要求反转,这样会破坏原链表结构)
- 2.用栈的方法,利用栈先进后出的特点,实现逆序打印的效果
stack.java
下面是栈的实例
import java.util.Stack;
public class stack{
public static void main(String[] args){
Stack<String> stack = new Stack<>();
stack.add("James");
stack.add("Kboe");
stack.add("Jordan");
while(stack.size() > 0){
System.out.println(stack.pop());
}
}
}
用栈的方法实现从头到尾打印单链表
public static void reversePrint(PersonNode head) {
if (head.next == null)
return;
//创建一个栈,将各个结点压入栈
Stack<PersonNode> stack = new Stack<PersonNode>();
PersonNode current = head.next;
while (current != null) {
stack.push(current);
current = current.next;
}
while (stack.size() > 0) {
System.out.println(stack.pop());
}
}
测试
reversePrint(list.getHead());
单链表(Java--尚硅谷)的更多相关文章
- 数据结构——单链表java简易实现
巩固数据结构 单链表java实现 单链表除了表尾 每个几点都有一个后继 结点有数据和后继指针组成 通过构建表头和表尾(尾部追加需要)两个特殊几点 实现单链表的一些操作,代码如下 package co ...
- 单链表---java实现
单链表优点:1.不需要预先给出元素个数. 2.单链表插入删除时不需要移动数据元素. 单链表缺点:1.每个节点有指针,空间利用率低. 2.单链表不支持随机读取数据. Node.java package ...
- 单链表Java实现
近期在复习基本数据结构,本文是单链表的Java实现,包含对单链表的实现插入删除查找遍历等.最后还实现了单链表的逆置. 实现了多项式相加,多项式相乘. 原文章及完整源码在这里 http://binhua ...
- 尚硅谷全套课件整理:Java、前端、大数据、安卓、面试题
目录 Java 尚硅谷 IT 精英计划 JavaSE 内部学习笔记.pdf 尚硅谷 Java 基础实战之银行项目.pdf 尚硅谷 Java 技术之 JDBC.pdf 尚硅谷 Java 技术之 Java ...
- 尚硅谷Java高级笔记
尚硅谷Java高级笔记 idea的使用: 一些小区别: 其他细节参考idea配置pdf 多线程: 基本概念: 多线程的优点: 何时需要多线程: 线程的创建和使用: 创建多线程的第一种方式: /** * ...
- Java实现单链表的各种操作
Java实现单链表的各种操作 主要内容:1.单链表的基本操作 2.删除重复数据 3.找到倒数第k个元素 4.实现链表的反转 5.从尾到头输出链表 6.找到中间节点 7.检测链表是否有环 8.在 ...
- JAVA 链表操作:单链表和双链表
主要讲述几点: 一.链表的简介 二.链表实现原理和必要性 三.单链表示例 四.双链表示例 一.链表的简介 链表是一种比较常用的数据结构,链表虽然保存比较复杂,但是在查询时候比较便捷,在多种计算机语言都 ...
- 数组、单链表和双链表介绍 以及 双向链表的C/C++/Java实现
概要 线性表是一种线性结构,它是具有相同类型的n(n≥0)个数据元素组成的有限序列.本章先介绍线性表的几个基本组成部分:数组.单向链表.双向链表:随后给出双向链表的C.C++和Java三种语言的实现. ...
- 用Java实现单链表的基本操作
笔试题中经常遇到单链表的考题,下面用java总结一下单链表的基本操作,包括添加删除节点,以及链表转置. package mars; //单链表添加,删除节点 public class ListNode ...
- java单链表代码实现
用惯了C++,java写起来果然不太爽...不废话了,上代码... package javaInnerclassDemo; class Link{ class Node{ private String ...
随机推荐
- 基于Flink构建全场景实时数仓
目录: 一. 实时计算初期 二. 实时数仓建设 三. Lambda架构的实时数仓 四. Kappa架构的实时数仓 五. 流批结合的实时数仓 实时计算初期 虽然实时计算在最近几年才火起来,但是在早期也有 ...
- PYTHON IDLE同时运行两个PY,互不影响
1.打开指定文件夹 2.右击a1.py,用IDLE打开(idle1) 3.右击a2.py,用IDLE打开(idle2) 4.则分别按F5运行,则两个互不影响. 5.再用IDLE1打开新程序,用IDLE ...
- Docker单机网络实战
前言 Docker系列文章: 此篇是Docker系列的第八篇,大家一定要按照我做的Demo都手敲一遍,印象会更加深刻的,加油! 为什么要学习Docker Docker基本概念 Docker镜像基本原理 ...
- 论文阅读:LIC-Fusion: LiDAR-Inertial-Camera Odometry
本文提出了一种紧耦合的多传感器(雷达-惯导-相机)融合算法,将IMU测量.稀疏视觉特征.提取的激光点融合.提出的算法在时间和空间上对三个异步传感器进行在线校准,补偿校准发生的变化.贡献在于将检测和追踪 ...
- IO编程之IO流
Java的IO流是实现输入.输出的基础,它可以方便的实现数据的输入输出操作.在Java中把不同的输入输出源(键盘.文件.网络连接)抽象表述为流(stream).通过流可以使java程序使用相同的方式访 ...
- 前端开发入门到进阶第三集【js高度计算公式】
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/stri ...
- SpringBoot之了解自动配置原理
承接上文: import注解的使用: 作用:引用时,容器自动创建出这填写的类型的组件,默认组件的名字就是全类名. 往容器中添加两个组件 @Import({User.class, DBHelper.cl ...
- Html模板引擎Handlerbars使用demo
1.自定义demo <html> <head> <script src="./handlebars-v4.0.12.js"></scrip ...
- Beego和Vue的前后端分离跨域问题处理
VUE封装的请求头(注意请求头,跨域要用到) 路径 utils/mereq.js import request from '@/utils/request' import qs from 'qs' e ...
- [IOI2005]mea
IOI 读完题,感觉这个题并不是很难,那我是不是可以去IOI了: 最先考虑暴力,发现完全行不通,所以,我们考虑其他方法.突然发现:其实在确定 \(s_1\) 的时候,整个序列就可以确定了,所以我们考虑 ...