前言

其实在学习数据结构之前,我也是从来都没了解过这门课,但是随着工作的慢慢深入,之前学习的东西实在是不够用,并且太皮毛了。太浅,只是懂得一些浅层的,我知道这个东西怎么用,但是要优化、或者是解析,就不知道该咋弄了。比如JAVA 最有名的几个容器:

  • List
  • Set
  • MAP
  • Queue

这些都是涉及到有关数据结构的,以及一些简单的算法。排序、冒泡排序、二分法这些,都要涉及到时间复杂度、以及数据结构的知识,这门课,还是很重要的。

为了啥

其实数据结构,结构这个词,就是将我们原本的一些数据,按照某种结构放到一起,为了更加便利以及后期对于这些数据的利用。不能胡来,乱放一遭,那样整理起来很麻烦,并且不方便以后的二次利用。

平时使用的数据,要么是基本类型、要么就是引用类型、数组、这些就是最基本的。加入需要存一个比如层级结构的岗位,那普通的数组就没有办法了。

这里我们所涉及到的内容其实就是 数据的 结构

结构分类

数据结构的分类,到底有哪些呢,如何去理解他们,就是我们本节课的内容。这里我们将接触到线性表、树状图、图存储结构等

线性表

线性表其实和数组有些类似。我们都知道,所有数据的类型都可以通过

最基本的 数组 指针(引用类型) 这两种最基本的类型构造。

线性表可以细分为:

  • 顺序表
  • 链表
  • 队列

本节课就围绕线性表,将这几种类型依次解释清楚

顺序表

顺序表最常见的,当然就是数组(不等同数组),满足 一对一 何谓一对一呢,就是其里面存储的元素,他们的类型,都是存在相同类型的关系,并且紧挨着连接起来的。例如:

String [] array = new String[] {"a","b","c"};

类似于这种,除掉首元素和尾元素,每个元素前后都有相邻的元素。这样的我们就叫做顺序表

JAVA 里面我们知道最基本的List 接口,下面有一个 ArrayList

ArrayList 底层就是以一个数组,其就是一个顺序表。

基本操作

我这里全部以JAVA 为例。

    public ArrayList(int initialCapacity) {
if (initialCapacity > 0) {
this.elementData = new Object[initialCapacity];
} else if (initialCapacity == 0) {
this.elementData = EMPTY_ELEMENTDATA;
} else {
throw new IllegalArgumentException("Illegal Capacity: "+
initialCapacity);
}
}

通过这个构造函数,我们可以发现,传入一个指定的大小数,大于0,则指定基本数组的大小为传入大小。 虽然这个数组是支持自动扩容的,我们还是研究一下

add() 增加元素到尾部

    public boolean add(E e) {
ensureCapacityInternal(size + 1); // Increments modCount!!
elementData[size++] = e;
return true;
}

ensureCapacityInternal是对数组的监测,若大小不足以容纳,则扩容的机制

这里的增加元素其实很简单,就将元素放到size ,也就是容器当中元素数的位置,首次放入元素的时候,size 初始化就是0 而后自增,很简单

add() 插入元素


public void add(int index, E element) {
rangeCheckForAdd(index); ensureCapacityInternal(size + 1); // Increments modCount!!
System.arraycopy(elementData, index, elementData, index + 1,
size - index);
elementData[index] = element;
size++;
}

rangeCheckForAdd 检查插入位置有没有超过数组大小。则直接抛出异常

扩容后,将插入点后面的元素往后移动一个位置,通过System.arraycopy复制方式实现

查找指定下标 get()

    public E get(int index) {
rangeCheck(index); return elementData(index);
}

这个就不细说了,太简单了。

remove() 移除元素

    public E remove(int index) {
rangeCheck(index); modCount++;
E oldValue = elementData(index); int numMoved = size - index - 1;
if (numMoved > 0)
System.arraycopy(elementData, index+1, elementData, index,
numMoved);
elementData[--size] = null; // clear to let GC do its work return oldValue;
}

这个和上面指定位置插入一个元素刚好相反,把指定位置的元素移除掉后,后面的元素往前移动一位,而后将最后元素的位置进行清理。

总结

顺序表最大的特点是:查询快,因为是数组,直接下标出。插入和移除就比较慢了。因为要移动、复制数组,很麻烦

链表

在上面我们已经说过了,任何的数据类型都可以通过最基本的数组和指针构造。链表也不例外,相比于数组,数组则是定长的,不管存储的满否,都申请了一定大小的内存空间,而链表则不是,链表的空间是随用随申请,数据的位置相比于数组,其实不连续的,一般来说,需要在元素上指定下一个元素的指针,来达成链接关系。

每个元素上都有一块位置用于指向下一个元素(指针)

这里我画的不连续也就是为了表示元素的不连续性

    private static class Node<E> {
E item;
Node<E> next;
Node<E> prev; Node(Node<E> prev, E element, Node<E> next) {
this.item = element;
this.next = next;
this.prev = prev;
}
}

链表在JAVA 当中最具代表性的就是 LinkedList(双向链表),就是每个元素会带有它的上一个节点和下一个节点的指针,我们图上画出来的是单向链表。

add(E e)

    void linkLast(E e) {
final Node<E> l = last;
final Node<E> newNode = new Node<>(l, e, null);
last = newNode;
if (l == null)
first = newNode;
else
l.next = newNode;
size++;
modCount++;
}

新下挂一个节点的时候,将最后一个节点(null)保存到 l 下,然后构造出一个新节点,将本节点作为最后一个最后一个节点。

add(int Index,E e)

    void linkBefore(E e, Node<E> succ) {
// assert succ != null;
final Node<E> pred = succ.prev;
final Node<E> newNode = new Node<>(pred, e, succ);
succ.prev = newNode;
if (pred == null)
first = newNode;
else
pred.next = newNode;
size++;
modCount++;
}

这里两个参数,E 表示将要插入的元素,

两边链表断开,new 一个新的节点,连接即可。

get(i) 查找

    Node<E> node(int index) {
// assert isElementIndex(index); if (index < (size >> 1)) {
Node<E> x = first;
for (int i = 0; i < index; i++)
x = x.next;
return x;
} else {
Node<E> x = last;
for (int i = size - 1; i > index; i--)
x = x.prev;
return x;
}
}

链表的查找指定下标就比较费时了。需要一个个遍历。其实是很麻烦的。

remove(i)

    E unlink(Node<E> x) {
// assert x != null;
final E element = x.item;
final Node<E> next = x.next;
final Node<E> prev = x.prev; if (prev == null) {
first = next;
} else {
prev.next = next;
x.prev = null;
} if (next == null) {
last = prev;
} else {
next.prev = prev;
x.next = null;
} x.item = null;
size--;
modCount++;
return element;
}

移除一个指定位置的节点,这个其实和增加一个节点时候其实是类似的。将上下节点对于这个节点的引用进行修改即可。

小结

链表还是比较适合于快速增加、删除、不适合于索引。因为需要全盘遍历

栈 Stack

堆还是按照数组为基础实现的,只不过它是一个半开的数组,怎么理解这个半开的数组呢,如图,就好像一个瓶子一样,往里面丢元素,先进后出原则

入栈 push

将一个元素加入的栈里面,此时的元素是最外层的一个元素,此时执行出栈命令,则这个元素会被删除并返回

出栈 pop

删除此堆栈顶部的对象,并将该对象作为此函数的值返回。

查看 peek

通过 peek 查看当前栈顶的元素,只是查看,并不执行删除

队列 Queue

队列遵循先进先出原则

队列还提供额外的插入,提取和检查操作。 这些方法中的每一种都有两种形式:如果操作失败,则抛出一个异常,另一种返回一个特殊值( null或false ,具体取决于操作)。

这里使用 ArrayBlockingQueue 以数组实现的阻塞队列

BlockingQueue<String> strings = new ArrayBlockingQueue<String>(2);

一个有限的blocking queue由数组支持。 这个队列排列元素FIFO(先进先出)。 队列的头部是队列中最长的元素。 队列的尾部是队列中最短时间的元素。 新元素插入队列的尾部,队列检索操作获取队列头部的元素。

这是一个经典的“有界缓冲区”,其中固定大小的数组保存由生产者插入的元素并由消费者提取。 创建后,容量无法更改

入队 add()/offer()/put()

add 和 offer 都可以将元素加入到队列中。但是add 在超过队列容量的时候会抛出异常,offer 则会返回false

而put 操作则会在队列没有空间的时候阻塞,直到队列有空间执行

出队 poll()/take()

  • poll检索并删除此队列的头,如果此队列为空,则返回 null 。
  • take 在没有元素的时候则会阻塞

小结

通过本小结,我们已经学习到了最基本的线性表,而线性表又包含哪些呢

  • 顺序表 ArrayList
  • 链表 LinkedList
  • 栈 Stack
  • 队列 Queue

下一节我们将继续学习有关于字符串、数组、广义表等内容

参考:

http://c.biancheng.net/view/3352.html

数据结构 1 线性表详解 链表、 栈 、 队列 结合JAVA 详解的更多相关文章

  1. 【Java】 大话数据结构(2) 线性表之单链表

    本文根据<大话数据结构>一书,实现了Java版的单链表. 每个结点中只包含一个指针域的链表,称为单链表. 单链表的结构如图所示: 单链表与顺序存储结构的对比: 实现程序: package ...

  2. 【Java】 大话数据结构(3) 线性表之静态链表

    本文根据<大话数据结构>一书,实现了Java版的静态链表. 用数组描述的链表,称为静态链表. 数组元素由两个数据域data和cur组成:data存放数据元素:cur相当于单链表中的next ...

  3. [C++]数据结构:线性表之(单)链表

    一 (单)链表 ADT + Status InitList(LinkList &L) 初始化(单)链表 + void printList(LinkList L) 遍历(单)链表 + int L ...

  4. [数据结构-线性表1.2] 链表与 LinkedList<T>(.NET 源码学习)

    [数据结构-线性表1.2] 链表与 LinkedList<T> [注:本篇文章源码内容较少,分析度较浅,请酌情选择阅读] 关键词:链表(数据结构)    C#中的链表(源码)    可空类 ...

  5. javascript实现数据结构:线性表--线性链表(链式存储结构)

    上一节中, 线性表的顺序存储结构的特点是逻辑关系上相邻的两个元素在物理位置上也相邻,因此可以随机存取表中任一元素,它的存储位置可用一个简单,直观的公式来表示.然后,另一方面来看,这个特点也造成这种存储 ...

  6. [数据结构 - 第3章] 线性表之单链表(C++实现)

    一.类定义 单链表类的定义如下: #ifndef SIGNALLIST_H #define SIGNALLIST_H typedef int ElemType; /* "ElemType类型 ...

  7. 玩转C线性表和单向链表之Linux双向链表优化

    前言: 这次介绍基本数据结构的线性表和链表,并用C语言进行编写:建议最开始学数据结构时,用C语言:像栈和队列都可以用这两种数据结构来实现. 一.线性表基本介绍 1 概念: 线性表也就是关系户中最简单的 ...

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

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

  9. 数据结构之线性表(python版)

    数据结构之线性表(python版) 单链表 1.1  定义表节点 # 定义表节点 class LNode(): def __init__(self,elem,next = None): self.el ...

随机推荐

  1. Perl:只是把“^”作为匹配的单字:只是匹配每一行的开头 $lines =~ s/^/file_4_ex_ch7.txt: /gm;

    Perl:只是把“^”作为匹配的单字:只是匹配每一行的开头                     $lines =~ s/^/file_4_ex_ch7.txt: /gm;

  2. 怎么保证RabbitMQ和kafuka集群的高可用性?

    rabbitMQ有三种模式:单机模式,普通集群模式,镜像集群模式 RabbitMQ的高可用性   RabbitMQ是比较有代表性的,因为是基于主从做高可用性的,我们就以他为例子讲解第一种MQ的高可用性 ...

  3. java中的赋值

    java中的赋值使用符号“=”. 按照java编程思想的解释:它的意思是“取等号右边的值,把它复制给左边”. 当然左边必须是一个明确的,已命名的变量. 基本类型: int a=2; int b=3; ...

  4. Logarithmic transformation|Data transfer|MASS|Box-Cox

    数据转换(Data transfer) 方差分析的前提是方差齐性,可以使用transfer改变方差使得方差变齐.不正态和outlier. Logarithmic transformation使方差聚合 ...

  5. IE11阅读视图:带给你静心饕餮阅读大餐的片刻

    编者按:又到读书日,今天你挤出时间读书了吗?如今,越来越多人在习惯电子阅读,然而总难逃眼花缭乱的干扰信息.Internet Explorer 11新增阅读视图功能,一键开启,给你带给你静心饕餮阅读大餐 ...

  6. Qt QString与string的转换

    QString --> string QString.toStdString();   string --> QString QString::fromStdString(string);

  7. win10安装motionbuilder失败,怎么强力卸载删除注册表并重新安装

    一些搞设计的朋友在win10系统下安装motionbuilder失败或提示已安装,也有时候想重新安装motionbuilder的时候会出现本电脑windows系统已安装motionbuilder,你要 ...

  8. Python自动化之数据驱动(转载下自己的日常练习笔记)

    一.execl数据驱动 1.xlutils简要说明 导包:pip3 install xlutils 注意⚠️:xlutils在介些Execl文件的时候,只识别.xls后缀的文件,如果是.xlsx后缀的 ...

  9. Linux Ubuntu 查看IP

    一.第一种方式:ifconfig -a 1.打开命令终端:Crtl + Alt + T 2.执行命令:ifconfig -a 如果你没有安装net-tools工具,则出现如下显示: (没有找到'ifc ...

  10. C语言Windows程序设计—— 使用计时器

    传统意义上的计时器是指利用特定的原理来测量时间的装置, 在古代, 常用沙漏.点燃一炷香等方式进行粗略的计时, 在现代科技的带动下, 计时水平越来越高, 也越来越精确, 之所以需要进行计时是在很多情况下 ...