(原创)用Java实现链表结构对象:单向无环链表
转载请注明本文出处:http://www.cnblogs.com/Starshot/p/6918569.html
链表的结构是由一个一个节点组成的,所谓链,就是每个节点的头尾连在一起。而单向链表就是:每个节点包含了当前节点的值和下一个节点引用。双向链表就是每个节点包含了当前节点的值和上下两个节点的引用。相对于数组结构,链表的增删效率会更加高。
这边文章主要讲怎么用Java实现一个简单的链表结构:单向无环链表。以及实现一些数据处理的方法。
首先,新建一个节点类(本次例子中的节点值都是字符串类型):
- public class Node {
- private String value;
- private Node next;
- public Node(String value){
- this.value=value;
- }
- public Node(){
- }
- public String getValue(){
- return value;
- }
- public Node getnext(){
- return next;
- }
- public void setValue(String value){
- this.value=value;
- }
- public void setNext(Node next){
- this.next=next;
- }
- }
然后再建一个链表类:
- public class MyLinkedTable {
- Node head=new Node();
- //向链表中增加值
- public boolean add(String str){
- Node node=new Node(str);
- if(head.getnext()==null){
- head.setNext(node);;
- return true;
- }
- Node tmp=head;
- while(tmp.getnext()!=null){
- tmp=tmp.getnext();
- }
- tmp.setNext(node);
- return true;
- }
- //已知某个节点,然后删除该节点
- public boolean delete(Node node){
- Node next=node.getnext();
- Node nextAddress=next.getnext();
- String nextValue=next.getValue();
- node.setNext(nextAddress);
- node.setValue(nextValue);
- return true;
- }
- //删除第index个节点,index从1开始
- public boolean delete(int index){
- if(index>length()||index<1){
- return false;
- }
- int i=1;
- Node temp=head.getnext();
- while(i<index){
- temp=temp.getnext();
- i++;
- }
- Node next=temp.getnext();
- Node nextAddress=next.getnext();
- String nextValue=next.getValue();
- temp.setNext(nextAddress);
- temp.setValue(nextValue);
- return true;
- }
- //获得第index个节点的值
- public String get(int index){
- if(index>length()||index<1){
- return null;
- }
- int i=1;
- Node temp=head.getnext();
- while(i<index){
- temp=temp.getnext();
- i++;
- }
- return temp.getValue();
- }
- //获取链表里面数据的长度,也就是插入了多少个值。
- public int length(){
- if(head.getnext()==null){
- return 0;
- }
- int i=1;
- Node temp=head.getnext();
- while((temp=temp.getnext())!=null){
- i++;
- }
- return i;
- }
- //反转链表
- public void reverseTable(){
- Node node1=head.getnext();
- if(node1==null){
- return;
- }
- Node preNode=null;
- Node curNode=node1;
- while(true){
- Node origNextNode=curNode.getnext();
- curNode.setNext(preNode);
- if(origNextNode==null){
- head.setNext(curNode);
- break;
- }
- preNode=curNode;
- curNode=origNextNode;
- }
- }
- //获取中间链表
- public String getMid(){
- Node slowPointer=this.head.getnext(),fastPointer=this.head.getnext();
- while(fastPointer!=null&&fastPointer.getnext()!=null&&fastPointer.getnext().getnext()!=null){
- slowPointer=slowPointer.getnext();
- fastPointer=fastPointer.getnext().getnext();
- }
- return slowPointer.getValue();
- }
- //删除重复链表
- public void deleteDuplicateNode(){
- Node nodeCur=head.getnext();
- if(nodeCur==null) return;
- while(nodeCur.getnext()!=null){
- Node lastNode=nodeCur;
- Node compareNode=lastNode.getnext();
- while(compareNode!=null){
- if(nodeCur.getValue().equals(compareNode.getValue())){
- lastNode.setNext(compareNode.getnext());
- compareNode=compareNode.getnext();
- }else{
- lastNode=compareNode;
- compareNode=compareNode.getnext();
- }
- }
- nodeCur=nodeCur.getnext();
- }
- }
- }
需要说明一下,这个链表结构是以head为起始节点,head指向插入的第一个节点(反转后就是最后一个),head本身的value一直都是空的,它只作为起始节点入口存在,不参与节点的计数,所以链表的节点总数(插入数据总数)是不包括head在内的。
接下来对某些方法进行说明:
1.链表反转
- //链表翻转就是把原来的1>2>3>4变成4>3>2>1,所以原来在最前面的节点要变成在最后面,最后面的要变成最前面
- //就是head的地址引用要变成最后一个节点的,原来head后的第一个节点的地址引用要由第二的节点变为NULL,原来第二个节点的地址引用要由第三个节点变成第一个节点
- //以此类推,一直翻转到最后一个。然后把head的地址引用变成最后一个。
- //在这个链表里面,head的位置是一直不变的,它永远是最前面,在head之后的节点才开始翻转。
- public void reverseTable(){
- Node node1=head.getnext();
- if(node1==null){
- return;
- }
- Node preNode=null;
- Node curNode=node1;
- while(true){
- Node origNextNode=curNode.getnext();
- curNode.setNext(preNode);
- if(origNextNode==null){
- head.setNext(curNode);
- break;
- }
- preNode=curNode;
- curNode=origNextNode;
- }
- }
2.获得中间节点:
- //通过用快慢指针来找到中间节点,快指针的速度为慢指针的两倍,慢指针一次走一个节点,快指针一次走两个节点,
- //当快指针走到尽头时,慢指针刚好为中间值,当快指针走到倒数第二个节点时,慢指针为上中位数。
- //fastPointer!=null用来判断链表表内除head外有没有其它节点,fastPointer.getnext()!=null判断是否为最后一个节点,
- //fastPointer.getnext().getnext()!=null判断是否为倒数第二个节点。
- public String getMid(){
- Node slowPointer=this.head.getnext(),fastPointer=this.head.getnext();
- while(fastPointer!=null&&fastPointer.getnext()!=null&&fastPointer.getnext().getnext()!=null){
- slowPointer=slowPointer.getnext();
- fastPointer=fastPointer.getnext().getnext();
- }
- return slowPointer.getValue();
- }
3.删除具有相同值的节点(重复节点):
- //删除具有相同值的节点
- //基本原理是用第一个节点的值和第二个节点值比较,然后和第三个节点比较,以此类推。此时第一个节点为当前节点nodeCur,第二第三以及之后的节点为比较节点compareNode
- //一轮比较完毕后,第二个节点就变成nodeCur,之后那些节点就是compareNode
- //如果遇到有相同的值的节点,就将该节点的上个节点的next值为该节点的下个节点:lastNode.setNext(compareNode.getnext()),此时该节点就在链表里失去引用了,就相当于删除了。
- //所以需要lastNode引用来保存当前比较节点的上一个节点
- public void deleteDuplicateNode(){
- Node nodeCur=head.getnext();
- if(nodeCur==null) return;
- while(nodeCur.getnext()!=null){
- Node lastNode=nodeCur;
- Node compareNode=lastNode.getnext();
- while(compareNode!=null){
- if(nodeCur.getValue().equals(compareNode.getValue())){
- lastNode.setNext(compareNode.getnext());
- compareNode=compareNode.getnext();
- }else{
- lastNode=compareNode;
- compareNode=compareNode.getnext();
- }
- }
- nodeCur=nodeCur.getnext();
- }
- }
写好之后可以测试一下:
- public class Test {
- public static void main(String[] args) {
- MyLinkedTable m=new MyLinkedTable();
- m.add("1");
- m.add("2");
- m.add("3");
- m.add("4");
- m.add("5");
- m.add("6");
- for(int i=1;i<=m.length();i++){
- System.out.println(i+":"+m.get(i));
- }
- System.out.println("length:"+m.length());
- m.reverseTable();
- for(int i=1;i<=m.length();i++){
- System.out.println(i+":"+m.get(i));
- }
- System.out.println("length:"+m.length());
- m.delete(2);
- for(int i=1;i<=m.length();i++){
- System.out.println(i+":"+m.get(i));
- }
- System.out.println("length:"+m.length());
- System.out.println(m.getMid());
- }
- }
运行输出如下:
1:1
2:2
3:3
4:4
5:5
6:6
length:6
1:6
2:5
3:4
4:3
5:2
6:1
length:6
1:6
2:4
3:3
4:2
5:1
length:5
3
还有测试删除重复节点的:
- public class Test2 {
- public static void main(String[] args) {
- MyLinkedTable m=new MyLinkedTable();
- m.add("1");
- m.add("2");
- m.add("3");
- m.add("2");
- m.add("2");
- m.add("6");
- for(int i=1;i<=m.length();i++){
- System.out.println(i+":"+m.get(i));
- }
- System.out.println("length:"+m.length());
- m.deleteDuplicateNode();
- for(int i=1;i<=m.length();i++){
- System.out.println(i+":"+m.get(i));
- }
- System.out.println("length:"+m.length());
- }
- }
运行输出:
1:1
2:2
3:3
4:2
5:2
6:6
length:6
1:1
2:2
3:3
4:6
length:4
以上就是用java实现单向无环链表的详细过程和解释,如果不妥之处,欢迎指出。
(原创)用Java实现链表结构对象:单向无环链表的更多相关文章
- JavaScript实现单向链表结构
参考资料 一.什么是链表结构? 1.1.简介 链表和数组一样, 可以用于存储一系列的元素, 但是链表和数组的实现机制完全不同,链表中的元素在内存不是连续的空间,链表的每个元素由一个存储元素本身(数据) ...
- 【二叉树->链表】二叉树结构转双向线性链表结构(先序遍历)
二叉树存储结构属于非线性链表结构,转化成线性链表结构,能简化操作和理解.然而由非线性转线性需要对整个树遍历一次,不同的遍历方式转化结果页不一样.下面以先序为例. 方法一: 递归法.递归遍历二叉树,因为 ...
- Linux内核--链表结构(一)
一.前言 Linux内核链表结构是一种双向循环链表结构,与传统的链表结构不同,Linux内核链表结构仅包含前驱和后继指针,不包含数据域.使用链表结构,仅需在结构体成员中包含list_head*成员就行 ...
- 《Java数据结构》链表结构(单向链表,双向链表)
单向链表(单链表)是链表的一种,其特点是链表的链接方向是单向的,对链表的访问要通过顺序读取从头部开始:链表是使用指针进行构造的列表:又称为结点列表,因为链表是由一个个结点组装起来的:其中每个结点都有指 ...
- JAVA基础——链表结构之单链表
链表:一种数据存储结构.学链表首先要搞懂数组,按朋友的话说,数组和链表的关系就相当于QQ2008和QQ2009. 除非要通过索引频繁访问各个数据,不然大多数情况下都可以用链表代替数组. 链表部分主要要 ...
- Atitit 跨平台异常处理(2)--------异常转换 -----java c# js异常对象结构比较and转换
Atitit 跨平台异常处理(2)--------异常转换 -----java c# js异常对象结构比较and转换 { "@type":"java.lang.Runti ...
- 由PHP实现单向链表引发的对象赋值,对象传参,链表操作引发的一系列问题
2019年2月25日14:21:13 测试版本php 5.4 ,5.6,7.0,7.2 代码请看: https://www.cnblogs.com/zx-admin/p/10373866.html 1 ...
- 面试题:JVM在Java堆中对对象的创建、内存结构、访问方式
一.对象创建过程 1.检查类是否已被加载 JVM遇到new指令时,首先会去检查这个指令参数能否在常量池中定位到这个类的符号引用,检查这个符号引用代表的类是否已被加载.解析.初始化,若没有,则进行类加载 ...
- java实现链表结构详细代码
一.数据准备 1. 定义节点 2. 定义链表 1.数据部分 2.节点部分 class DATA //数据节点类型 { String key; String name; int age; } cla ...
随机推荐
- xshell配色Solarized Dark
转自:xshell配色Solarized Dark [Solarized_Dark] text(bold)= magenta(bold)=6c71c4 text= white(bold)=fdf6e3 ...
- 谈谈一些有趣的CSS题目(十五)-- 谈谈 CSS 关键字 initial、inherit 和 unset
开本系列,谈谈一些有趣的 CSS 题目,题目类型天马行空,想到什么说什么,不仅为了拓宽一下解决问题的思路,更涉及一些容易忽视的 CSS 细节. 解题不考虑兼容性,题目天马行空,想到什么说什么,如果解题 ...
- STM32、Cortex-A、Cortex-R、Cortex-M、SecurCore
STM32是就是基于Cortex-M3这个核生产的CPU. arm7是arm公司推出的以V4指令集设计出来的arm核--其代表的芯片有s3c44b0 arm9是arm公司推出的以V5指令集设计出来的a ...
- 保证Android后台不被杀死的几种方法
由于各种原因,在开发Android应用时会提出保证自己有一个后台一直运行的需求,如何保证后台始终运行,不被系统因为内存低杀死,不被任务管理器杀死,不被软件管家等软件杀死等等还是一个比较困难的问题.网上 ...
- HTTP长连接、短连接使用及测试
概念 HTTP短连接(非持久连接)是指,客户端和服务端进行一次HTTP请求/响应之后,就关闭连接.所以,下一次的HTTP请求/响应操作就需要重新建立连接. HTTP长连接(持久连接)是指,客户端和服务 ...
- Native App和Web App 的差异
开发者们都知道在高端智能手机系统中有两种应用程序:一种是基于本地(操作系统)运行的APP:一种是基于高端机的浏览器运行的WebApp,本文将主要讲解后者. WebApp与Native App有何区别呢 ...
- Angularjs快速入门(二)
说说上一节的例子,$scope 我们没有创建这个对象,直接绑定就能获取里面的对象,这种风格遵循了一种叫迪米特法则的设计模式. 然后angular还有一种很强大的功能叫“指令”. 就是你可以吧模板编写成 ...
- Excel基本操作1
Excel的基本操作之二,录入及快速填充.不足之处,欢迎补充
- C语言精要总结-指针系列(一)
考虑到指针内容繁多,这里将指针作为一个系列,从简入繁,一点一点深挖并掌握这C语言的精华.初步计划如下 此文为指针系列第一篇: C语言精要总结-指针系列(一) 内存与地址 我们可以把内存看做一排连续的房 ...
- base 镜像 - 每天5分钟玩转容器技术(10)
上一节我们介绍了最小的 Docker 镜像,本节讨论 base 镜像. base 镜像有两层含义: 不依赖其他镜像,从 scratch 构建. 其他镜像可以之为基础进行扩展. 所以,能称作 base ...