主要讲述几点:

一、链表的简介

二、链表实现原理和必要性

三、单链表示例

四、双链表示例

 

一、链表的简介 

  链表是一种比较常用的数据结构,链表虽然保存比较复杂,但是在查询时候比较便捷,在多种计算机语言都相应的应用,链表有多种类别,文章针对单链表和双链表进行分析。链表中数据就像被一个链条串联一起,轻易的可以实现数据的访问。

二、链表实现原理和必要性

  这里只分析单链表和双链表。链表的实现过程是有些许复杂的,但是会带来许多好处。比如现在网购时代到来,商家发快递一般会将商品包装在盒子里并写上地址信息,快递公司就可以通过盒子上的信息找到买家,商品完整到达。如果没有盒子的保护,有可能在途中商品受损。而链表就好比那个写了地址信息的盒子,既保护了商品信息,同时又写好了物流信息。链表之中存在一个HEAD节点,类似“火车头”,只要找到相应HEAD节点,就可以对链表进行操作。此次分析中,HEAD节点只是做数据头,不保存有效数据。

  单链表的实现原理如图:

  

  双链表实现原理:

  

三、单链表示例  

ICommOperate<T> 接口操作类:

package LinkListTest;
import java.util.Map;
public interface ICommOperate<T> { public boolean insertNode(T node) ;
public boolean insertPosNode(int pos, T node) ;
public boolean deleteNode(int pos) ;
public boolean updateNode(int pos, Map<String, Object> map) ;
public T getNode(int pos, Map<String, Object> map) ;
public void printLink() ;
}

单链表节点:

package LinkListTest;
// 单连表节点
public class SNode {
private String data;
private SNode nextNode;
public SNode() {
}
public SNode(String data) {
this.data = data;
this.nextNode = new SNode();
} public String getData() {
return data;
}
public void setData(String data) {
this.data = data;
}
public SNode getNextNode() {
return nextNode;
}
public void setNextNode(SNode nextNode) {
this.nextNode = nextNode;
}
@Override
public String toString() {
return "SNode [data=" + data + "]";
}
}

单链接操作类:

package LinkListTest;
import java.util.HashMap;
import java.util.Map;
public class SingleLinkList implements ICommOperate<SNode>{
private SNode head = new SNode("HEAD") ; // 公共头指针,声明之后不变
private int size = 0 ;
public int getSize() {
return this.size;
} /*
* 链表插入,每次往末端插入
* */
@Override
public boolean insertNode(SNode node) {
boolean flag = false ;
SNode current = this.head ;
if( this.size==0 ){ // 空链表
this.head.setNextNode(node) ;
node.setNextNode(null) ;
}else{ // 链表内节点
while( current.getNextNode()!=null ){
current = current.getNextNode() ;
}
current.setNextNode(node) ;
node.setNextNode(null) ;
}
this.size++ ;
flag = true ; return flag;
} /*
* 插入链表指定位置pos,从1开始,而pos大于size则插入链表末端
* */
@Override
public boolean insertPosNode(int pos, SNode node){
boolean flag = true;
SNode current = this.head.getNextNode() ; if( this.size==0 ){ // 空链表
this.head.setNextNode(node) ;
node.setNextNode(null) ;
this.size++ ;
}else if( this.size<pos ){ // pos位置大于链表长度,插入末端
insertNode(node) ;
}else if( pos>0 && pos<=this.size) { // 链表内节点
// 1、找到要插入pos位置节点和前节点
int find = 0;
SNode preNode = this.head; // 前节点
SNode currentNode = current; // 当前节点
while( find<pos-1 && currentNode.getNextNode()!=null ){
preNode = current ; // 前节点后移
currentNode = currentNode.getNextNode() ; // 当前节点后移
find++ ;
}
// System.out.println(preNode);
// System.out.println(currentNode);
// 2、插入节点
preNode.setNextNode(node);
node.setNextNode(currentNode);
this.size++ ;
System.out.println("节点已经插入链表中");
}else{
System.out.println("位置信息错误");
flag = false ;
} return flag;
} /*
* 指定链表的节点pos,删除对应节点。方式:找到要删除节点的前后节点,进行删除。从1开始
* */
@Override
public boolean deleteNode(int pos) {
boolean flag = false;
SNode current = this.head.getNextNode() ;
if( pos<=0 || pos>this.size || current==null ){
System.out.println("位置信息错误或链表无信息");
}else{
// 1、找到要删除节点的前后节点
int find = 0;
SNode preNode = this.head; // 前节点
SNode nextNode = current.getNextNode(); // 后节点
while( find<pos-1 && nextNode.getNextNode()!=null ){
preNode = current ; // 前节点后移
nextNode = nextNode.getNextNode() ; // 后节点后移
find++ ;
}
// System.out.println(preNode);
// System.out.println(nextNode); // 2、删除节点
preNode.setNextNode(nextNode);
System.gc();
this.size-- ;
flag = true ;
} return flag;
} /*
* 指定链表的节点pos,修改对应节点。 从1开始
* */
@Override
public boolean updateNode(int pos, Map<String, Object> map) {
boolean flag = false ;
SNode node = getNode(pos, map); // 获得相应位置pos的节点
if( node!=null ){
String data = (String) map.get("data") ;
node.setData(data);
flag = true ;
}
return flag;
} /*
* 找到指定链表的节点pos,从1开始
* */
@Override
public SNode getNode(int pos, Map<String, Object> map) {
SNode current = this.head.getNextNode() ;
if( pos<=0 || pos>this.size || current==null ){
System.out.println("位置信息错误或链表不存在");
return null;
}
int find = 0 ;
while( find<pos-1 && current!=null ){
current = current.getNextNode() ;
find++ ;
}
return current;
} /*
* 打印链表
* */
@Override
public void printLink() {
int length = this.size ;
if( length==0 ){
System.out.println("链表为空!");
return ;
}
SNode current = this.head.getNextNode() ;
int find = 0 ;
System.out.println("总共有节点数: " + length +" 个");
while( current!=null ){
System.out.println("第 " + (++find) + " 个节点 :" + current);
current=current.getNextNode() ;
}
} public static void main(String[] args) {
SingleLinkList sll = new SingleLinkList() ;
SNode node1 = new SNode("节点1");
SNode node2 = new SNode("节点2");
SNode node3 = new SNode("节点3");
SNode node4 = new SNode("节点4");
SNode node5 = new SNode("节点5");
SNode node6 = new SNode("插入指定位置");
sll.insertPosNode(sll.getSize()+1, node1) ;
sll.insertPosNode(sll.getSize()+1, node2) ;
sll.insertPosNode(sll.getSize()+1, node3) ;
sll.insertPosNode(sll.getSize()+1, node4) ;
sll.insertPosNode(sll.getSize()+1, node5) ; // sll.insertNode(node1);
// sll.insertNode(node2);
// sll.insertNode(node3);
// sll.insertNode(node4);
// sll.insertNode(node5); System.out.println("*******************输出链表*******************");
sll.printLink(); System.out.println("*******************获得指定链表节点*******************");
int pos = 2 ;
System.out.println("获取链表第 "+pos+" 个位置数据 :"+sll.getNode(pos, null)); System.out.println("*******************向链表指定位置插入节点*******************");
int pos1 = 2 ;
System.out.println("将数据插入第 "+pos1+" 个节点:");
sll.insertPosNode(pos1, node6) ;
sll.printLink(); System.out.println("*******************删除链表指定位置节点*******************");
int pos2 = 2 ;
System.out.println("删除第 "+pos2+" 个节点:");
sll.deleteNode(pos2) ;
sll.printLink(); System.out.println("*******************修改链表指定位置节点*******************");
int pos3 = 2 ;
System.out.println("修改第 "+pos3+" 个节点:");
Map<String, Object> map = new HashMap<>() ;
map.put("data", "this is a test") ;
sll.updateNode(pos3, map) ;
sll.printLink();
}
}

四、双链表示例

ICommOperate<T> 接口操作类:

package LinkListTest;
import java.util.Map;
public interface ICommOperate<T> {
public boolean insertNode(T node) ;
public boolean insertPosNode(int pos, T node) ;
public boolean deleteNode(int pos) ;
public boolean updateNode(int pos, Map<String, Object> map) ;
public T getNode(int pos, Map<String, Object> map) ;
public void printLink() ;
}

双链表节点:

package LinkListTest;
// 双连表节点
public class DNode {
private DNode priorNode;
private String data;
private DNode nextNode; public DNode(){
}
public DNode(String data) {
this.priorNode = new DNode() ;
this.data = data ;
this.nextNode = new DNode() ;
} public DNode getPriorNode() {
return priorNode;
}
public void setPriorNode(DNode priorNode) {
this.priorNode = priorNode;
} public String getData() {
return data;
}
public void setData(String data) {
this.data = data;
} public DNode getNextNode() {
return nextNode;
}
public void setNextNode(DNode nextNode) {
this.nextNode = nextNode;
} @Override
public String toString() {
return "DNode [data=" + data + "]";
}
}

双链表实现类:

package LinkListTest;
import java.util.HashMap;
import java.util.Map;
public class DoubleLinkList implements ICommOperate<DNode>{
private DNode head = new DNode("HEAD");
private int size = 0 ;
public int getSize() {
return this.size;
} /*
* 链表插入,每次往末端插入
* */
@Override
public boolean insertNode(DNode node) {
boolean flag = false; DNode current = this.head ;
if( this.size==0 ){ // 空链表
this.head.setNextNode(node) ;
node.setPriorNode(this.head);
node.setNextNode(null) ;
}else{ // 链表内节点
while( current.getNextNode()!=null ){
current = current.getNextNode() ;
}
current.setNextNode(node);
node.setNextNode(null);
node.setPriorNode(current);
}
this.size++ ;
flag = true ; return flag;
} /*
* 插入链表指定位置pos,从1开始,而pos大于size则插入链表末端
* */
@Override
public boolean insertPosNode(int pos, DNode node) {
boolean flag = true; DNode current = this.head.getNextNode() ;
if( this.size==0){ // 链表为空
this.head.setNextNode(node) ;
node.setNextNode(null) ;
node.setPriorNode(this.head);
this.size++ ;
}else if( pos>this.size ){ // pos位置大于链表长度,插入末端
insertNode(node) ;
}else if( pos>0 && pos<=this.size ){ // 链表内节点
// 1、找到要插入位置pos节点,插入pos节点当前位置
int find = 0;
while( find<pos-1 && current.getNextNode()!=null ){
current = current.getNextNode() ;
find++ ;
}
// 2、插入节点
if( current.getNextNode()==null ){ // 尾节点
node.setPriorNode(current);
node.setNextNode(null);
current.setNextNode(node);
} else if( current.getNextNode()!=null ) { //中间节点
node.setPriorNode(current.getPriorNode());
node.setNextNode(current);
current.getPriorNode().setNextNode(node);
current.setPriorNode(node);
}
this.size++ ;
}else{
System.out.println("位置信息错误");
flag = false ;
} return flag;
} /*
* 指定链表的节点pos,删除对应节点,从1开始
* */
@Override
public boolean deleteNode(int pos) {
boolean flag = false;
DNode current = this.head.getNextNode() ;
if( pos<=0 || pos>this.size || current==null ){
System.out.println("位置信息错误或链表不存在");
}else{
// 1、找到要删除位置pos节点
int find = 0;
while( find<pos-1 && current.getNextNode()!=null ){
current = current.getNextNode() ;
find++ ;
}
// 2、删除节点
if( current.getNextNode()==null ){ // 尾节点
current.getPriorNode().setNextNode(null) ;
} else if( current.getNextNode()!=null ) { //中间节点
current.getPriorNode().setNextNode(current.getNextNode()) ;
current.getNextNode().setPriorNode(current.getPriorNode()) ;
}
System.gc();
this.size-- ;
flag = true ;
}
return flag;
} /*
* 指定链表的节点pos,修改对应节点。 从1开始
* */
@Override
public boolean updateNode(int pos, Map<String, Object> map) {
boolean flag = false ;
DNode node = getNode(pos, map);
if( node!=null ){
String data = (String) map.get("data") ;
node.setData(data);
flag = true ;
}
return flag;
} /*
* 找到指定链表的节点pos,从1开始
* */
@Override
public DNode getNode(int pos, Map<String, Object> map) {
DNode current = this.head.getNextNode() ;
if( pos<=0 || pos>this.size || current==null ){
System.out.println("位置信息错误或链表不存在");
return null;
}
int find = 0 ;
while( find<pos-1 && current!=null ){
current = current.getNextNode() ;
find++ ;
}
return current;
} /*
* 打印链表
* */
@Override
public void printLink() {
int length = this.size ;
if( length==0 ){
System.out.println("链表为空!");
return ;
}
DNode current = this.head.getNextNode() ;
int find = 0 ;
System.out.println("总共有节点数: " + length +" 个");
while( current!=null ){
System.out.println("第 " + (++find) + " 个节点 :" + current);
current=current.getNextNode() ;
}
} public static void main(String[] args) {
DoubleLinkList dll = new DoubleLinkList() ;
DNode node1 = new DNode("节点1");
DNode node2 = new DNode("节点2");
DNode node3 = new DNode("节点3");
DNode node4 = new DNode("节点4");
DNode node5 = new DNode("节点5");
DNode node6 = new DNode("插入指定位置");
dll.insertPosNode(10, node1) ;
dll.insertPosNode(10, node2) ;
dll.insertPosNode(10, node3) ;
dll.insertPosNode(10, node4) ;
dll.insertPosNode(10, node5) ;
// dll.insertNode(node1);
// dll.insertNode(node2);
// dll.insertNode(node3);
// dll.insertNode(node4);
// dll.insertNode(node5); System.out.println("*******************输出链表*******************");
dll.printLink(); System.out.println("*******************获得指定链表节点*******************");
int pos = 2 ;
System.out.println("获取链表第 "+pos+" 个位置数据 :"+dll.getNode(pos, null)); System.out.println("*******************向链表指定位置插入节点*******************");
int pos1 = 2 ;
System.out.println("将数据插入第"+pos1+"个节点:");
dll.insertPosNode(pos1, node6) ;
dll.printLink(); System.out.println("*******************删除链表指定位置节点*******************");
int pos2 = 7 ;
System.out.println("删除第"+pos2+"个节点:");
dll.deleteNode(pos2) ;
dll.printLink(); System.out.println("*******************修改链表指定位置节点*******************");
int pos3 = 2 ;
System.out.println("修改第"+pos3+"个节点:");
Map<String, Object> map = new HashMap<>() ;
map.put("data", "this is a test") ;
dll.updateNode(pos3, map) ;
dll.printLink();
}
}

JAVA 链表操作:单链表和双链表的更多相关文章

  1. 数组、单链表和双链表介绍 以及 双向链表的C/C++/Java实现

    概要 线性表是一种线性结构,它是具有相同类型的n(n≥0)个数据元素组成的有限序列.本章先介绍线性表的几个基本组成部分:数组.单向链表.双向链表:随后给出双向链表的C.C++和Java三种语言的实现. ...

  2. 《程序员代码面试指南》第二章 链表问题 在单链表和双链表中删除倒数第K个节点

    题目 在单链表和双链表中删除倒数第K个节点 java代码 /** * @Description:在单链表和双链表中删除倒数第K个节点 * @Author: lizhouwei * @CreateDat ...

  3. C# 数据结构 - 单链表 双链表 环形链表

    链表特点(单链表 双链表) 优点:插入和删除非常快.因为单链表只需要修改Next指向的节点,双链表只需要指向Next和Prev的节点就可以完成插入和删除操作. 缺点:当需要查找某一个节点的时候就需要一 ...

  4. Java双链表

    一.概述 二.英雄类 class HeroNode { //值域 public int id; public String name; public String nickName; //指针域 pu ...

  5. Linux 底下使用C语言的 单链表 ,双链表,二叉树 读取文件,并排序

    直接上代码 单链表Linux读文件排序: 双链表Linux读取文件排序: 二叉树LinuX读取文件并排序:

  6. C语言版本:双链表的实现

    Dlist.h #ifndef __DLIST_H__ #define __DLIST_H__ #include<cstdio> #include<malloc.h> #inc ...

  7. Python与数据结构[0] -> 链表/LinkedList[1] -> 双链表与循环双链表的 Python 实现

    双链表 / Doubly Linked List 目录 双链表 循环双链表 1 双链表 双链表和单链表的不同之处在于,双链表需要多增加一个域(C语言),即在Python中需要多增加一个属性,用于存储指 ...

  8. 算法 - 链表操作思想 && case

    算法 - 链表操作题目套路 前面这一篇文章主要讲链表操作时候的实操解决方式,本文从本质讲解链表操作的元信息,学完后,再也不怕链表操作题目了. 1.链表的基本操作 链表的基本操作无外乎插入,删除,遍历 ...

  9. JAVA 链表操作:循环链表

    主要分析示例: 一.循环链表简述 二.单链表循环链表 三.双链表循环链表 一.循环链表简述 循环链表即链表形成了一个循环的结构,尾节点不再指向NULL,而是指向头节点HEAD,此时判定链表的结束是尾节 ...

随机推荐

  1. 測試大型資料表的 Horizontal Partitioning 水平切割

    FileGroup 檔案群組 :一個「資料庫(database)」可對應一或多個 FileGroup,一個 FileGroup 可由一或多個 file (.ndf) 構成. FileGroup 可讓 ...

  2. 使用Ado.net执行SP很慢,而用SSMS执行很快

    今天遇到一个问题,有用户反应,在site上打开报表,一直loading,出不来结果. 遇到这种问题,我立刻simulate用户使用Filter Condition,问题repro,看来不是偶然事件,通 ...

  3. Sql Server系列:索引基础

    1 索引概念 索引用于快速查找在某个列中某个特定值的行,不使用索引,数据库必须从第1条记录开始读完整个表,知道找出需要的行.表越大,查询数据所花费的时间越多.如果表中查询的列有索引,数据库能快速到达一 ...

  4. 解读sencha touch移动框架的核心架构(二)

    本来这行要详解Ext.extend的,但是发现网站有很详细的,那么就跳过去吧 为保持一个系列的分析,还是先搬过来吧,下章开始分析Ext4.0的新架构 在Java中,我们在实现继承的时候存在下面几个事实 ...

  5. 使用pudb调试python

    本博客主要用于讲解如何使用pudb进行python调试: 1.安装 sudo pip install pudb pip list查看安装结果: 2.使用 测试程序: #!/usr/bin/env py ...

  6. php面向对象基础

    1.类 由众多对象抽象出来的 它包含了对象通用的特性 2.对象 一切皆对象 它是由实例化出来的 例: 求两个圆之间阴影的面积 <!DOCTYPE html PUBLIC "-//W3C ...

  7. 写在最前面 - 每天5分钟玩转 OpenStack(1)

    <每天5分钟玩转 OpenStack>是一个 OpenStack 教程,这是第 1 篇. 这个教程有下面两个特点: 系统讲解 OpenStack 从架构到各个组件:从整体到细节逐一讨论 重 ...

  8. EntityFramework之Log(五)

    关于日志 属性日志 DbContext.Database.Log 属性被设置为一个委托,该委托能接受带有一个字符串参数的任何方法,最主要的是,通过设置它到 TextWriter 的 Write 方法将 ...

  9. 设计窘境:来自 Repository 的一丝线索,Domain Model 再重新设计

    写在前面 阅读目录: 疑惑解读 设计窘境 一幅图的灵感 为嘛还是你-Repository 后记 上一篇<No zuo no die:DDD 应对具体业务场景,Domain Model 重新设计& ...

  10. Java内存模型深度解析:锁--转

    原文地址:http://www.codeceo.com/article/java-memory-5.html 锁的释放-获取建立的happens before 关系 锁是java并发编程中最重要的同步 ...