线性表(1)

线性表即“把所有数据按照顺序(线性)的存储结构方式,存储在物理空间”。

线性表又分为

  • 顺序表
  • 链表
  1.  单向链表
  2. 双向链表

一、顺序表 

数据依次存储在连续的物理空间中,就比如数组。

顺序表存储数据时,会提前申请一整块足够大小的内存,然后将数据依次存储起来,元素的存储空间在内存中是连续存在的。

顺序表的优点:

  •  内存地址连续,数组元素进行遍历时,速度快。
  • 根据下标,查找指定位置的元素时,速度快。(时间复杂度为  O(1)  )

顺序表的缺点:

  • 长度固定,使用前需要预估长度。
  • 插入删除元素时,时间复杂度相对较高。(时间复杂度为  O(n)  )

二、链表 

与顺序表不同,链表不限制数据的物理存储位置,使用链表存储的数据元素,其物理存储位置是随机的。

链表中每个数据的存储都由以下两部分组成:

1G4212457-2.gif

  1. 数据元素本身,其所在的区域称为数据域;
  2. 指向直接后继元素的指针,所在的区域称为指针域

 链表的优点:

  • 使用链表结构,不需要提前预估长度,可以克服数组需要预先知道数据长度的缺点
  • 链表使用不连续的内存空间,可以充分利用计算机内存空间,实现灵活的内存动态管理

链表的缺点: 

  • 链表相比于数组会占用更多的空间,因为链表中每个节点中,除了存放元素本身,还有存放指向其他节点的指针
  • 不能随机读取元素(RandomAccess
  • 遍历和查找元素的速度比较慢

链表又分为:

  1. 单向链表
  2. 双向链表
  3. 循环链表
  4. 双向循环链表

(1)单向链表

1、单向链表的节点定义

//单向链表的节点定义
static class Node{
    E item;//元素值
    Node next;//后继节点(指针域)
    public Node(E data){
        this.item=data;
    }
}

2、链表的遍历 

//	链表的遍历
	@Override
	public String toString() {
		StringJoiner sj=new StringJoiner("->");
		//从首元素开始遍历
		for(Node n=first;n!=null;n=n.next) {
			sj.add(n.item.toString());
		}
		return sj.toString();
	}

3、头插法 

其主要思想为在原链表的头部添加一个新元素,我们只需要创建出一个新元素,然后让这个新元素的后继节点指向原来的头结点。

public class Linked{
    //头节点
    Node first;

    //尾节点
    Node last;

    //添加新元素(头插法)
    public  void addfirst(E item){
        final Node newfirst=new Node(item);//创建出新的头节点
		final Node f=first;//将原来的头节点存储起来
		first=newfirst;//将新创建的新的节点赋给头节点
		if(f==null) {
			last=newfirst;//如果这个链表为空时,它的头节点和尾节点都为所要添加的这个新节点
		}else {
			newfirst.next=f;//如果链表不为空,则新节点的后继节点指向原来的头节点
		}
    }
}

线性表(1)_第1张图片

4、尾插法

其主要思想是在原链表的尾部添加一个节点,使得原来的尾节点的后继节点指向新添加的这个新节点

//	尾插法
	public void addlast(E item) {
		final Node newlast=new Node(item);
		final Node l=last;//将原来的尾节点存储起来
		last=newlast;//将创建的新的节点赋给尾节点
		if(l==null) {
			first=newlast;
		}else {
			l.next=newlast;//如果链表不为空,则原尾结点的后继节点指向新节点
		}
	}

线性表(1)_第2张图片

5、删除头结点

其主要思想是删除旧的头节点,再将旧的头节点的后继节点作为新的头节点

//	删除头结点
	public void delfirst() {
		final Node f=first;//先保存头节点
		final Node next=f.next;//拿到头节点的后继节点
		f.item=null;//删除旧的头节点
		f.next=null;
		first=next;//设置新的头节点
		if(next==null) {
			//单向链表为空,则尾结点为null
			last=null;
		}
	}

线性表(1)_第3张图片

(2)循环链表

循环链表 其实是一种特殊的单链表,和单链表不同的是循环链表的尾结点不是指向 null,而是指向链表的头结点。

线性表(1)_第4张图片

(3) 双向链表

线性表(1)_第5张图片

1、双向链表的节点类

static class Node{
		E item;//数据域
		Node prev;//前驱节点
		Node next;//后继节点
		
//		构造方法
		public Node(E element,Node prev,Node next) {
			this.item=element;
			this.next=next;
			this.prev=prev;
		}

2、双向链表的头插法 

双向链表在头插法时,其新加入的节点的前驱节点指向null,后继节点指向原来的头节点

    Node first;
	Node last;
	int size=0;
	
	public  void addfirst(E item) {
		final Node f=first;
		final Node newfirst=new Node(item,null,f);
		first=newfirst;
		if(f==null) {
			last=newfirst;
		}else {
			newfirst.next=f;
		}
		size++;
	}

线性表(1)_第6张图片

3、双向链表的尾插法

双向链表的尾插法,其新加入的节点前驱节点指向原尾结点,后继节点指向null

public void addlast(E item) {
		final Node l=last;
		final Node newlast=new Node(item,l,null);
		last=newlast;
		if(l==null) {
			first=newlast;
		}else {
			l.next=newlast;
		}
		size++;
	}

线性表(1)_第7张图片

4、删除头节点 

// 删除链表头节点
public void removeFirst() {
    // 获取头结点
    final Node f = first;

    // 获取头结点的下一个节点
    final Node next = f.next;

    // 清空头结点
    f.item = null;
    f.next = null; // help GC

    // 设置链表的头结点
    first = next;

    if (next == null)
        last = null;
    else
        next.prev = null;
    
    // 链表长度自减
    size--;
}

线性表(1)_第8张图片 

5、 删除尾结点

// 删除链表尾节点
public void removeLast() {
    // 获取尾节点
    final Node l = last;

    // 获取尾节点的“上一个元素”
    final Node prev = l.prev;

    // 清空尾节点
    l.item = null;
    l.prev = null; // help GC

    // 设置链表的尾节点
    last = prev;

    if (prev == null)
        first = null;
    else
        prev.next = null;
    
    // 链表长度自减
    size--;
}

(4)双向循环链表

双向循环链表 最后一个节点的 next 指向 head,而 head 的 prev 指向最后一个节点,构成一个环。

线性表(1)_第9张图片

你可能感兴趣的:(算法)