顺序表示和链式表示的比较:

1.读写方式:顺序表可以顺序存取,也可以随机存取;链表只能从表头顺序存取元素;

2.逻辑结构与物理结构:顺序存储时,逻辑上相邻的元素其对应的物理存储位置也相邻;链式存储时,逻辑上相邻的元素,其物理存储位置则不一定相邻;

3.查找、插入和删除操作:

  按值查找,当线性表在无序的情况下,两者的时间复杂度均为o(n);而当顺序表有序时,可采用折半查找,此时时间复杂度为o(log n);

  按位查找,顺序表支持随机访问,时间复杂度为o(1);而链表的平均时间复杂度为o(n)。

  顺序表的插入和删除操作平均需要移动半个表长的元素;链表的插入、删除操作时,只需修改相关节点的指针即可。

4.空间分配:顺序存储一般是基于静态存储分配,一单存储空间装满就不能扩充;链式存储的节点空间只有在需要的时候申请分配,只要内存有足够的空间即可分配。

顺序表示的实现:

public class ArrayList<E> {
final int defaultSize=0;
int maxSize; //线性表的最大长度
int length; //线性表当前长度
Object[] arrayList; //存储线性表的数组 /*
* 构造函数
*/
public ArrayList(int size){
initList(size);
} //按给定size初始化顺序表
public void initList(int size) {
if(size < 0){
throw new RuntimeException("数组大小错误:" + size);
}else{
this.maxSize= size;
this.length=0;
this.arrayList = new Object[size];
}
} //表长
public int length() {
return length;
} //按值查找
public int locateElem(Object e) {
for(int i=0 ;i<length;i++){
if(arrayList[i] == e){
return i;
}
}
return -1;
} //按位查找
public Object getElem(int i) {
if(i<0 || i>=length ){
throw new RuntimeException("参数出错:"+i);
}
if(length ==0){
throw new RuntimeException("顺序表为空");
}
return arrayList[i];
} //插入
public void insert(int i, Object e) {
if(i<0 || i>length+1 ){
throw new RuntimeException("新元素插入位置有误:"+i);
}
if(i >= maxSize){
throw new RuntimeException("顺序表已满,不能再插入新的元素");
}
for(int j=length;j<i; j--){
arrayList[j]=arrayList[j-1];
}
arrayList[i]=e;
length++;
} //删除
public void delete(int i, Object e) {
if(i<0 || i > length-1){
throw new RuntimeException("需删除元素位置有误:"+i);
}
if(length == 0){
throw new RuntimeException("顺序表为空,不能删除元素");
}
for(int j=i;j<length-1;j++){
arrayList[j] = arrayList[j+1];
}
arrayList[length-1]="";
length--;
} //判空
public boolean isEmpty() {
return length==0 ? true : false;
} //删除顺序表
public void destroyList() {
this.arrayList=null;
this.length=0;
this.maxSize=0;
}
}

单链表表示的实现:

class Node<E>{
E e; //数据
Node<E> next; //下一个节点 Node(){} Node(E e){
this.e = e;
this.next = null;
}
} public class LinkedList<E> {
private Node<E> header = null; //头结点
int size=0; //链表大小 public LinkedList() {
this.header = new Node<E>();
} //得到链表的长度
public int length() {
return size;
} //按值查找节点,返回该节点的位置
public int locateElem(E e) {
Node<E> temp = header;
int count = 1;
while(temp.next != null && temp.e != e){
temp = temp.next;
count ++;
}
return count;
} //找到第index个位置的节点
public Node<E> getNode(int index) {
if(index > size || index < 0){
throw new RuntimeException("索引值有错:" + index);
}
Node<E> temp = new Node<E>();
temp = header;
int count=1;
while(count != index){
temp = temp.next;
count ++;
}
return temp;
} //尾添加
public boolean addToLast(E e) {
if(size == 0){
header.e = e;
}else{
Node<E> newnode = new Node<E>(e); //根据需要添加的内容封装为节点
Node<E> last = getNode(size); //得到最后一个节点
last.next = newnode;
}
size ++;
return true;
} //头添加
public boolean addTofirst(E e) {
if(size == 0){
header.e = e;
}else{
Node<E> newnode = new Node<E>(e); //根据需要添加的内容封装为节点
newnode.next = header.next;
header.next = newnode;
}
size ++;
return true;
} //插入到第index个的位置
public boolean insert(int index, E e) {
Node<E> newnode = new Node<E>(e); //根据需要添加的内容封装为节点
Node<E> cnode = getNode(index-1); //得到第index-1个节点
newnode.next = cnode.next;
cnode.next = newnode;
size++;
return true;
} //删除第index个节点
public boolean delete(int index) {
Node<E> prinode = getNode(index-1); //得到被删除的节点的前一个节点
Node<E> delnode = prinode.next; //得到被删除的节点
prinode.next = delnode.next;
size --;
return true;
} //判空
public boolean isEmpty() {
return size==0 ? true : false;
} public void destroyList() {
header = null;
size = 0;
} //输出
public String toString(){
StringBuilder s = new StringBuilder("[");
Node<E> temp = header;
for(int i=0; i < size;i++){
s.append(temp.e.toString()+" ");
temp = temp.next;
}
s.append("]");
return s.toString();
}
}

双链表表示的实现

class TNode<E>{
E e;
TNode<E> prior, next; TNode(){}
TNode(E e){
this.e = e;
prior = null;
next = null;
}
} public class DoubleLinkedList<E> {
private TNode<E> header = null; //头结点
int size=0; //链表大小 public DoubleLinkedList(){
this.header = new TNode<E>();
} //尾添加
public boolean addToLast(E e) {
if(size == 0){
header.e = e;
}else{
TNode<E> TNode = new TNode<E>(e); //根据需要添加的内容封装为节点
TNode<E> last = getNode(size); //得到最后一个节点
last.next = TNode;
TNode.prior=last;
}
size ++;
return true;
} //找到第index个位置的节点
public TNode<E> getNode(int index){
if(index > size || index < 0){
throw new RuntimeException("索引值有错:" + index);
}
TNode<E> temp = new TNode<E>();
temp = header;
int count =1;
while(count != index){
temp = temp.next;
count ++;
}
return temp;
} //插入到第index个的位置
public boolean insert(int index,E e){
TNode<E> TNode = new TNode<E>(e);
TNode<E> cnode = getNode(index-1); //找到第index-1个位置的节点
TNode.next=cnode.next;
TNode.prior = cnode;
cnode.next.prior = TNode;
cnode.next = TNode;
size++;
return true;
} //删除第index个节点
public boolean delete(int index){
TNode<E> delnode = getNode(index);
delnode.prior.next=delnode.next;
delnode.next.prior= delnode.prior;
size--;
return true;
}
}

Java实现线性表-顺序表示和链式表示的更多相关文章

  1. 数据结构Java实现05----栈:顺序栈和链式堆栈

    一.堆栈的基本概念: 堆栈(也简称作栈)是一种特殊的线性表,堆栈的数据元素以及数据元素间的逻辑关系和线性表完全相同,其差别是线性表允许在任意位置进行插入和删除操作,而堆栈只允许在固定一端进行插入和删除 ...

  2. 数据结构Java实现03----栈:顺序栈和链式堆栈

    一.堆栈的基本概念: 堆栈(也简称作栈)是一种特殊的线性表,堆栈的数据元素以及数据元素间的逻辑关系和线性表完全相同,其差别是线性表允许在任意位置进行插入和删除操作,而堆栈只允许在固定一端进行插入和删除 ...

  3. 算法与数据结构(一) 线性表的顺序存储与链式存储(Swift版)

    温故而知新,在接下来的几篇博客中,将会系统的对数据结构的相关内容进行回顾并总结.数据结构乃编程的基础呢,还是要不时拿出来翻一翻回顾一下.当然数据结构相关博客中我们以Swift语言来实现.因为Swift ...

  4. 数据结构导论 四 线性表的顺序存储VS链式存储

    前几章已经介绍到了顺序存储.链式存储 顺序存储:初始化.插入.删除.定位 链式存储:初始化.插入.删除.定位 顺序存储:初始化 strudt student{ int ID://ID char nam ...

  5. 线性表的顺序存储和链式存储的实现(C)

    //线性表的顺序存储 #include <stdio.h>typedef int DataType;#define MaxSize 15//定义顺序表typedef struct { Da ...

  6. 线性表的顺序存储和链式存储c语言实现

    一.线性表的顺序存储 typedef int ElemType;typedef struct List { ElemType *data;//动态分配 ,需要申请空间 int length; }Lis ...

  7. c数据结构 -- 线性表之 复杂的链式存储结构

    复杂的链式存储结构 循环链表 定义:是一种头尾相接的链表(即表中最后一个结点的指针域指向头结点,整个链表形成一个环) 优点:从表中任一节点出发均可找到表中其他结点 注意:涉及遍历操作时,终止条件是判断 ...

  8. 数据结构----线性表顺序和链式结构的使用(c)

    PS:在学习数据结构之前,我相信很多博友也都学习过一些语言,比如说java,c语言,c++,web等,我们之前用的一些方法大都是封装好的,就java而言,里面使用了大量的封装好的方法,一些算法也大都写 ...

  9. 线性表——顺序表的实现与讲解(C++描述)

    线性表 引言 新生安排体检,为了 便管理与统一数据,学校特地规定了排队的方式,即按照学号排队,谁在前谁在后,这都是规定好的,所以谁在谁不在,都是非常方便统计的,同学们就像被一条线(学号)联系起来了,这 ...

随机推荐

  1. jdbc 1.0

    1. jdbc : java数据库连接技术 2.主要用到的类及接口 Class Driver ManagerDriver Connection Statement PreparedStatement ...

  2. Dsyy的第一篇博文~

    2017-08-07  周一  晴热热热热热 咳咳,很多人看到dsyy第一反应是什么意思?当然是大神媛媛!显然不是些(diao)(si)yy.(da)(si)yy...的别义,咋有点此地无银三百两的感 ...

  3. 桥接,NAT,Host Only的区别

    桥接,NAT,Host Only的区别   一.Brigde——桥接 :默认使用VMnet0fish批注:只要在虚拟机中将IP设对,即使宿主机的IP是错的,也可以通信.但是如此物理网卡被禁用了,则不能 ...

  4. script 执行的三种方式

    <script>: 脚本的获取和执行是同步的.此过程中页面被阻塞,停止解析. <script defer = "defer">:脚本的获取是异步的,执行是同 ...

  5. PHP内置标准类

    PHP内置标准类 php语言内部,有“很多现成的类”,其中有一个,被称为“内置标准类”. 这个类“内部”可以认为什么都没有,类似这样: class  stdclass{ } 其作用,可以用于存储一些临 ...

  6. HDU4055_Number String

    题目告诉你在一个排列中,相邻两个数的大小关系.问你排列可能有多少种情况. DP. f[i][j]表示将i个数按照前面i-1个大小关系排列且最后一个数位j的排列数有多少个. 这样对于新加入的一个数i+1 ...

  7. jquery 集合注意点

  8. Day20-初识Ajax

    想要实现的功能:点击提交以后,让数据发到后台进行验证,但是页面不刷新.悄悄提交用Ajax. 那么返回的字符串怎么样展示到前端HTML页面呢?可以在HTML中写个标签,定义一个选择器. 利用$('#id ...

  9. ubuntu adduser

    ubuntu adduser 添加用户 hexan 到目录 /home/hexan 下.$ sudo adduser hexan -home /home/hexan 添加一个普通用户adduser [ ...

  10. gitlab配置自动同步

    如果需要同步到生产环境,请做额外处理,如自动化测试,测试通过再同步. <?php $project = trim($_GET['project']); if (empty($project)) ...