JAVA单链表的实现-不带头结点且没有尾指针
本程序采用JAVA语言实现了线性表的链式实现。首先定义了线性表的接口ListInterface,然后LList类实现了ListInterface完成了链表的实现。
本实现中,链表是不带表头结点的,且有一个指针始终指向链表中的第一个元素,并没有定义尾指针。因此,每次向链表中插入新结点时需要遍历链表一次。
更详细的解释参考《数据结构与算法分析 JAVA语言描述第二版》Frank M. Carrano 著
ListInterface接口的定义如下:
public interface ListInterface<T> {
public boolean add(T newEntry);
public boolean add(int givenPosition, T newEntry);
public void clear();
public T remove(int givenPosition);
public boolean replace(int givenPosition, T newEntry);
public T getEntry(int givenPosition);
public boolean contains(T anEntry);
public int getLength();
public boolean isEmpty();
public void display();
}
具体的实现类LList定义如下:
public class LList<T> implements ListInterface<T>{
private Node firstNode;//指向第一个结点的指针,该链表是不带头结点的单链表
private int length;//表示单链表的长度
//Node类中不需要定义访问属性的get方法以及set方法,因为Node是内部类,内部类的属性可以直接在外部类中被访问
class Node{
//Node是内部类,其外部类中已经定义了T,故可以在这里使用通配符T
private T data;//结点的数据部分
private Node next;//结点的指针部分,指向下一个结点
//Node类中不需要默认构造器
public Node(T dataPortion){
data = dataPortion;
}
public Node(T dataPortion, Node nextNode){
data = dataPortion;
next = nextNode;
}
}
public LList(){
clear();
}
//获取链表中指定位置处的结点
private Node getNodeAt(int givenPosition){
assert (!isEmpty() && ((1 <= givenPosition) && (givenPosition <= length)));
Node currentNode = firstNode;
for(int counter = 1; counter < givenPosition; counter++){
currentNode = currentNode.next;
}
assert currentNode != null;
return currentNode;
}
@Override
public boolean add(T newEntry) {
// 将每个新结点插入到链表的末尾,通过getNodeAt()方法来获得最后一个元素的地址
Node newNode = new Node(newEntry);
if(isEmpty()){//插入第一个结点
firstNode = newNode;
}
else{//在其它位置插入结点
Node lastNode = getNodeAt(length);//这里每插入一个元素都需要遍历一次链表,代价较大
lastNode.next = newNode;
}
length++;
return true;
}
@Override
public boolean add(int givenPosition, T newEntry){//在指定位置处插入结点
boolean isSuccessful = true;
if(givenPosition >= 1 && givenPosition <= length + 1){
Node newNode = new Node(newEntry);
if(isEmpty() || givenPosition == 1){//在第一个位置处插入结点
newNode.next = firstNode;
firstNode = newNode;
}
else{//在其它位置插入结点
Node nodeBefore = getNodeAt(givenPosition - 1);
Node nodeAfter = nodeBefore.next;
nodeBefore.next = newNode;
newNode.next = nodeAfter;
}
length++;
}
else
isSuccessful = false;
return isSuccessful;
}
@Override
public final void clear() {//clear()在构造器中被调用了,所以用final修饰
firstNode = null;
length = 0;
}
@Override
public T remove(int givenPosition) {//删除指定位置处的结点
T result = null;
if((!isEmpty()) && ((givenPosition >= 1) && (givenPosition <= length))){
if(givenPosition == 1){//删除第一个位置处的结点
result = firstNode.data;
firstNode = firstNode.next;
}
else//删除表中其它位置结点
{
Node nodeBefore = getNodeAt(givenPosition - 1);
Node nodeToRemove = nodeBefore.next;
Node nodeAfter = nodeToRemove.next;
nodeBefore.next = nodeAfter;
result = nodeToRemove.data;
}
length--;
}
return result;
}
@Override
public boolean replace(int givenPosition, T newEntry) {//替换指定位置处结点的值
boolean isSuccessful = true;
if((!isEmpty()) && ((givenPosition >= 1) && (givenPosition <= length))){
Node desireNode = getNodeAt(givenPosition);
desireNode.data = newEntry;
}
else
isSuccessful = false;
return isSuccessful;
}
@Override
public T getEntry(int givenPosition) {//获取指定位置的结点的值
T result = null;
if((!isEmpty()) && ((givenPosition >= 1) && (givenPosition <= length))){
result = getNodeAt(givenPosition).data;
}
return result;
}
@Override
public boolean contains(T anEntry) {//判断链表中的结点是否包含某个值
boolean found = false;
Node currentNode = firstNode;
while(!found && currentNode != null){
if(currentNode.data.equals(anEntry)){
found = true;
break;
}
currentNode = currentNode.next;
}
return found;
}
@Override
public int getLength() {//获取链表的长度
return length;
}
@Override
public boolean isEmpty() {//判断链表是否为空
boolean result;
if(length == 0){
assert firstNode == null;
result = true;
}
else{
assert firstNode != null;
result = false;
}
return result;
}
@Override
public void display() {//遍历链表,显示链表中的每个结点的值
Node currentNode = firstNode;
while(currentNode != null){
System.out.println(currentNode.data);
currentNode = currentNode.next;
}
}
}
JAVA单链表的实现-不带头结点且没有尾指针的更多相关文章
- JAVA单链表的实现-不带头结点但带有尾指针
1,本程序实现了线性表的链式存储结构.实现的链表带有两个指针,一个始终指向链表中的第一个结点,另一个指针始终指向链表中的最后一个结点. 之所以设置尾指针,是因为,在插入元素到链表中的末尾时,可以通过尾 ...
- 【c++版数据结构】之循环单链表的实现(带头结点以及尾节点)
所实现的循环单链表的结构例如以下图所看到的: watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill ...
- java单链表常用操作
总结提高,与君共勉 概述. 数据结构与算法亘古不变的主题,链表也是面试常考的问题,特别是手写代码常常出现,将从以下方面做个小结 [链表个数] [反转链表-循环] [反转链表-递归] [查找链表倒数第K ...
- Java单链表反转 详细过程
版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/guyuealian/article/details/51119499 Java单链表反转 Java实 ...
- Java单链表反转图文详解
Java单链表反转图文详解 最近在回顾链表反转问题中,突然有一些新的发现和收获,特此整理一下,与大家分享 背景回顾 单链表的存储结构如图: 数据域存放数据元素,指针域存放后继结点地址 我们以一条 N1 ...
- 求单链表倒数第m个结点
问题:求单链表倒数第m个结点,要求不准求链表的长度,也不许对链表进行逆转 解:设置两个指针p和q,p.q指向第一个结点.让p先移动到链表的第m个结点,然后p和q同时向后移动,直到p首先到达尾结点.此时 ...
- java 单链表 练习
练习一下java单链表的简单习题 package com.test1; import java.util.Stack; public class SingleListDemo { /** * 返回单链 ...
- Java单链表简单实现* @version 1.0
package com.list; /** * 数据结构与算法Java表示 * @version 1.0 * @author 小明 * */ public class MyLinkedList { p ...
- 链表 | 递归删除不带头结点链表所有x元素
王道P37 T1 : 设计一个递归算法,删除不带头结点的单链表L中所有值为x的结点. 王道上的答案绝对是错的,我自己想了一个 函数主体 LinkList* del_x(LinkList* prior, ...
随机推荐
- 使用kindeditor来替换ecshop的fckeditor编辑器,让ecshop可以批量上传图片
老杨原创 kindeditor此编辑器可以让ecshop批量上传图片,可以插入代码,可以全屏编辑,可以插入地图.视频,进行更多word操作,设置字体. 步骤一:进入kindeditor的官网,http ...
- 一个简单的Oracle和 SQLSERVER 重建所有表索引的办法
1. SQLSERVER 使用微软自带的存储过程 exec sp_msforeachtable 'DBCC DBREINDEX(''?'')' 2. Oracle的办法: select 'alter ...
- Angular生成二维码
Installation - Angular 5+, Ionic NPM npm install angularx-qrcode --save Yarn yarn add angularx-qrcod ...
- loadrunner基础学习笔记三
运行时设置: 打开运行时设置:任务窗格中-选择回放-点击运行时设置按钮 1 重复执行次数:=2 2 步:控制迭代时间间隔 3 日志设置:指出要在运行测试期间记录的信息量 4 思考时间:可以在cont ...
- C#遍历类的属性,然后给其赋值
public class PP { public string a { get; set; } public string b { get; set; } public string c { get; ...
- day14 闭包
闭包的概念 必须要有函数嵌套,内部函数调用外部函数的变量 简单的例子 此种方法会导致每次使用内部函数inner的时候需要不断的调用外部函数. 结果导致外部函数的变量不断的被调用被释放,比较低效,相当于 ...
- MT【54】一道二次函数问题的几何意义
[Rather less, but better.]----卡尔·弗里德里希·高斯(1777-1855) (2016诸暨质检18)已知$f(x)=x^2-a|x-1|+b(a>0,b>-1 ...
- [TJOI2011]构造矩阵
考虑优化贪心,不回溯,对于每一位,你都判一下放0的话后面是否有解,用网络流判是否可以完美匹配就行了. 但这样时间复杂是错的,所以不必每次都重新建图,现在原来的图中看一下该行列是否已经匹配,若没有,则强 ...
- 详解基于朴素贝叶斯的情感分析及 Python 实现
相对于「 基于词典的分析 」,「 基于机器学习 」的就不需要大量标注的词典,但是需要大量标记的数据,比如: 还是下面这句话,如果它的标签是: 服务质量 - 中 (共有三个级别,好.中.差) ╮(╯-╰ ...
- luogu4187 [USACO18JAN]Stamp Painting (dp)
可以发现,只要存在连续k个相同的,这个情况就一定是合法情况 然而这个不太好算,我们算不存在k个相同的,然后用$m^n$把它减掉 设f[i]为前i个,没有连续k个的 显然$f[i]=m^i ,i< ...