目录
一、线性表
线性表是最基本、最简单、也是最常用的一种数据结构。一个线性表是n个具有相同特性的数据元素的有限序列。
前驱元素:若A元素在B元素的前面,则称A为B的前驱元素。
后继元素:若B元素在A元素的后面,则称B为A的后继元素。
线性表的特征:数据元素之间具有一种"一对一"的逻辑关系。
1.第一个数据元素没有前驱,这个数据元素被称为头节点。
2.最后一个数据元素没有后继,这个数据元素被称为尾节点。
3.除了第一个和最后一个数据元素外,其他数据元素有且仅有一个前驱和一个后驱。
线性表的分类:线性表中的数据可以是顺序存储,也可以是链式存储,按照数据的存储方式不同,可以把线性表分为顺序表和链表。
1、顺序表
顺序表是在计算机内存中以数组形式保存的线性表,线性表的顺序存储是指用一组地址连续的存储单元,依次存储线性表中的各个元素、使得线性表中在逻辑结构响应的数据元素存储在相邻的物理存储单元中,即通过数据元素存储的相邻关系来反映数据元素之间逻辑上的相邻关系。
顺序表API设计:
类名 | SequenceList<T> |
构造方法 | SequenceList(int capacity):创建容量为capacity的SequenceList对象 |
成员方法 | 1.public void clear():空置线性表 2.public boolean isEmpty():判断线性表是否为空,是返回true,否返回false 3.private int length():获取线性表中元素的个数 4.private T get(int i):读取并返回线性表中第i个元素的值 5.private void insert(int i,T t):在线性表的第i个元素之前插入一个值为t的元素 6.private void insert(T t):向线性表中插入一个t的元素 7.private T remove(int i):删除并返回线性表中第i个数据元素 8.private int indexOf(T t):返回线性表中首次出现的指定的数据元素的位序号,若不存在,则返回-1 |
成员变量 | 1.private T[] eles:存储元素的数组 2.private int N:当前线性表的长度 |
代码实现:
public class SequenceList<T> {
private T[] eles;
private int N;
// 构造方法
@SuppressWarnings("unchecked")
public SequenceList(int capacity) {
// 初始化数组
this.eles = (T[]) new Object[capacity];
// 初始化长度
this.N = 0;
}
// 空置线性表
public void clear() {
this.N = 0;
}
// 判断线性表是否为空,是返回true,否返回false
public boolean isEmpty() {
return N == 0;
}
// 获取线性表中元素的个数
public int length() {
return N;
}
// 读取并返回线性表中第i个元素的值
public T get(int i) {
return eles[i];
}
// 向线性表中插入一个t的元素
public void insert(T t) {
eles[N++] = t;
}
// 在线性表的第i个元素之前插入一个值为t的元素
public void insert(int i, T t) {
// 先把i索引处及其后面的元素依次向后移动一位
for (int index = N; index > i; index--) {
eles[index] = eles[index - 1];
}
// 再把t元素放到i索引处即可
eles[i] = t;
N++;
}
// 删除并返回线性表中第i个数据元素
public T remove(int i) {
// 记录索引i处的值,
T current = eles[i];
// 索引i后面元素依次向前移动一位
for (int index = i; index < N - 1; index++) {
eles[index] = eles[index + 1];
}
// 元素个数减一
N--;
return current;
}
// 返回线性表中首次出现的指定的数据元素的位序号,若不存在,则返回-1
public int indexOf(T t) {
for (int i = 0; i < N; i++) {
if (eles[i].equals(t)) {
return i;
}
}
return -1;
}
}
测试类:
public class SequenceListTest {
public static void main(String[] args) {
SequenceList<String> list = new SequenceList<>(10);
list.insert("张无忌");
list.insert("张翠山");
list.insert("张三丰");
list.insert("小昭");
list.insert(1, "赵敏");
System.out.println(list.get(1));
System.out.println(list.remove(0));
System.out.println(list.length());
list.clear();
System.out.println(list.length());
}
}
1.2、顺序表的遍历
在java代码中,遍历集合的方式一般都是用的是foreach循环,如果想让我们的SequenceList也能支持foreach循环,则需要做如下操作:
1、让SequenceList实现Iterable接口,重写Iterator方法
2、在SequenceList内部提供一个内部类SIterator,实现Iterator接口,重写hashNext方法和next方法
代码实现:
import java.util.Iterator;
public class SequenceList<T> implements Iterable<T> {
private T[] eles;
private int N;
// 构造方法
@SuppressWarnings("unchecked")
public SequenceList(int capacity) {
// 初始化数组
this.eles = (T[]) new Object[capacity];
// 初始化长度
this.N = 0;
}
// 空置线性表
public void clear() {
this.N = 0;
}
// 判断线性表是否为空,是返回true,否返回false
public boolean isEmpty() {
return N == 0;
}
// 获取线性表中元素的个数
public int length() {
return N;
}
// 读取并返回线性表中第i个元素的值
public T get(int i) {
return eles[i];
}
// 向线性表中插入一个t的元素
public void insert(T t) {
eles[N++] = t;
}
// 在线性表的第i个元素之前插入一个值为t的元素
public void insert(int i, T t) {
// 先把i索引处及其后面的元素依次向后移动一位
for (int index = N; index > i; index--) {
eles[index] = eles[index - 1];
}
// 再把t元素放到i索引处即可
eles[i] = t;
N++;
}
// 删除并返回线性表中第i个数据元素
public T remove(int i) {
// 记录索引i处的值,
T current = eles[i];
// 索引i后面元素依次向前移动一位
for (int index = i; index < N - 1; index++) {
eles[index] = eles[index + 1];
}
// 元素个数减一
N--;
return current;
}
// 返回线性表中首次出现的指定的数据元素的位序号,若不存在,则返回-1
public int indexOf(T t) {
for (int i = 0; i < N; i++) {
if (eles[i].equals(t)) {
return i;
}
}
return -1;
}
private class SIterator implements Iterator<T> {
private int cusor;
private SIterator() {
this.cusor = 0;
}
@Override
public boolean hasNext() {
return cusor < N;
}
@Override
public T next() {
return eles[cusor++];
}
}
@Override
public Iterator<T> iterator() {
return new SIterator();
}
}
测试类:
public class SequenceListTest {
public static void main(String[] args) {
SequenceList<String> list = new SequenceList<>(10);
list.insert("张无忌");
list.insert("张翠山");
list.insert("张三丰");
list.insert("小昭");
list.insert(1, "赵敏");
for (String string : list) {
System.out.println(string);
}
System.out.println("-----------------------------");
System.out.println(list.get(1));
System.out.println(list.remove(0));
System.out.println(list.length());
list.clear();
System.out.println(list.length());
}
}
1.3、顺序表的容量可变
考虑容器的容量伸缩性,其实就是改变存储数据元素的数组的大小,那我们需要考虑什么时候需要改变数组的大小?
1、添加元素时:添加元素时,应该检查当前数组的大小是否能容纳新的元素,如果不能容纳,则需要创建新的容量更大的数组,我们这里创建一个是原数组两倍容量的新数组存储元素。
2、移除元素时:移除元素时,应该检查当前数组的大小是否太大,比如正在用100个容量的数组存储10个元素,这样就会造成内存空间的浪费。应该创建一个更小的数组存储元素。如果我们发现数据元素的数量不足数组容量的1/4,则创建一个是原数组容量的1/2的新数组存储元素。
代码实现:
import java.util.Iterator;
public class SequenceList<T> implements Iterable<T> {
private T[] eles;
private int N;
// 构造方法
@SuppressWarnings("unchecked")
public SequenceList(int capacity) {
// 初始化数组
this.eles = (T[]) new Object[capacity];
// 初始化长度
this.N = 0;
}
// 空置线性表
public void clear() {
this.N = 0;
}
// 判断线性表是否为空,是返回true,否返回false
public boolean isEmpty() {
return N == 0;
}
// 获取线性表中元素的个数
public int length() {
return N;
}
// 读取并返回线性表中第i个元素的值
public T get(int i) {
return eles[i];
}
// 向线性表中插入一个t的元素
public void insert(T t) {
if (N == eles.length) {
resize(2 * eles.length);
}
eles[N++] = t;
}
// 在线性表的第i个元素之前插入一个值为t的元素
public void insert(int i, T t) {
if (N == eles.length) {
resize(2 * eles.length);
}
// 先把i索引处及其后面的元素依次向后移动一位
for (int index = N; index > i; index--) {
eles[index] = eles[index - 1];
}
// 再把t元素放到i索引处即可
eles[i] = t;
N++;
}
// 删除并返回线性表中第i个数据元素
public T remove(int i) {
// 记录索引i处的值,
T current = eles[i];
// 索引i后面元素依次向前移动一位
for (int index = i; index < N - 1; index++) {
eles[index] = eles[index + 1];
}
// 元素个数减一
N--;
if (N < eles.length / 4) {
resize(eles.length / 2);
}
return current;
}
// 返回线性表中首次出现的指定的数据元素的位序号,若不存在,则返回-1
public int indexOf(T t) {
for (int i = 0; i < N; i++) {
if (eles[i].equals(t)) {
return i;
}
}
return -1;
}
// 根据参数newSize,重置eles的大小
@SuppressWarnings("unchecked")
public void resize(int newSize) {
// 定义一个临时数组,指向原数组
T[] tmp = eles;
// 创建新数组
eles = (T[]) new Object[newSize];
// 把原数组的数据拷贝到新数组即可
for (int i = 0; i < N; i++) {
eles[i] = tmp[i];
}
}
private class SIterator implements Iterator<T> {
private int cusor;
private SIterator() {
this.cusor = 0;
}
@Override
public boolean hasNext() {
return cusor < N;
}
@Override
public T next() {
return eles[cusor++];
}
}
@Override
public Iterator<T> iterator() {
return new SIterator();
}
}
1.4、顺序表的时间复杂度
get(i):时间复杂度O(1);
insert(int i,T t);时间复杂度O(n);
remove(i);时间复杂度O(n);
由于顺序表的底层由数组实现,数组的长度是固定的,所以在操作的过程中涉及到了容器扩容操作。这样会导致顺序表在使用过程中的时间复杂度不是线性的,在某些需要扩充的节点处,耗时会突增,尤其是元素越多,这个问题越明显。
2、链表
天下事有难易乎?为之,则难者亦易矣;不为,则易者亦难矣。