??前面的话??
本篇文章带大家认识数据结构与算法基础,顺序表(动态),所谓的顺序表本质就是数组,它的特点是物理结构与逻辑结构都是连续的,最大的优点就是能够随机访问,最大的缺点是空间有限,就算扩容,也有可能存在大量的内存浪费。描述代码:Java。
📒博客主页:未见花闻的博客主页 🎉欢迎关注🔎点赞👍收藏??留言📝 📌本文由未见花闻原创,CSDN首发! 📆首发时间:🌴2021年11月10日🌴 ??坚持和努力一定能换来诗与远方! 💭参考书籍:📚《趣学数据结构》,📚《数据结构(C语言版)》,📚《趣学算法》 💬参考在线编程网站:🌐牛客网🌐力扣 博主的码云gitee,平常博主写的程序代码都在里面。 博主的github,平常博主写的程序代码都在里面。 🙏作者水平很有限,如果发现错误,一定要及时告知作者哦!感谢感谢!
🍓1.顺序表基本理论
🍇1.1线性表
线性表(linear list)是n个具有相同特性的数据元素的有限序列。 线性表是一种在实际中广泛使用的数据结构,常见的线性表:顺序表、链表、栈、队列、字符串… 线性表在逻辑上是线性结构,也就说是连续的一条直线。但是在物理结构上并不一定是连续的,线性表在物理上存储时,通常以数组和链式结构的形式存储。
本篇文章介绍的是最简单的数据结构——顺序表。
🍇1.2顺序表
🍅1.2.1概念与特点
顺序表是用一段物理地址连续的存储单元依次存储数据元素的线性结构,一般情况下采用数组存储。在数组上完成数据的增删查改。 顺序表一般可以分为:
- 静态顺序表:使用定长数组存储。
- 动态顺序表:使用动态开辟的数组存储。
静态顺序表适用于确定知道需要存多少数据的场景. 静态顺序表的定长数组导致N定大了,空间开多了浪费,开少了不够用. 相比之下动态顺序表更灵活, 根据需要动态的分配空间大小.
静态顺序表:
class StaticSeq {
public int Maxsize;
int[] elem = new int[Maxsize];
public int size;
public StaticSeq(int capacity) {
this.Maxsize = capacity;
}
}
动态顺序表:
class DynamicSeq {
public int[] elem;
public int size;
}
本篇文章将介绍动态顺序表增删查改的实现,静态顺序表不多赘述!
🍅1.2.2顺序表实现接口
public void initSeqList(){
}
public void display() {
}
public void add(int pos, int data) {
}
public boolean isFull() {
}
public boolean isEmpty() {
}
public boolean contains(int toFind) {
}
public int search(int toFind) {
}
public void seqAdd(int value) {
}
public int getPos(int pos) {
}
public void setPos(int pos, int value) {
}
public void remove(int toRemove) {
}
public int size() {
}
public void clear() {
}
🍓2.动态顺序表实践
🍇2.1动态顺序表基本结构
class DynamicSeq {
public int[] elem;
public int size;
}
🍇2.2初始化
顺序表的本质是数组,要使用顺序表必须有一段初始的空间,我初始化顺序表时,为顺序表开辟了10个数据空间。
public void initSeqList(){
this.elem = new int[10];
}
每次使用新的顺序表还需要自己构造对象调用初始化方法,太麻烦了,我们让顺序表在执行构造方法阶段就完成顺序表的初始化。
public class SeqList {
public int[] elem;
public int size;
public SeqList() {
this.initSeqList();
}
}
🍇2.3顺序插入数据
在最后一个有效元素后插入一个数据,当然,在此之前需要判断顺序表有没有满,如果满了需要先扩容再插入。判断顺序表满没满很简单,就看他的有效元素个数是否与数组长度相等,如果相等就代表顺序表满了。
public boolean isFull() {
return this.size == this.elem.length;
}
public void seqAdd(int value) {
if (this.elem == null) {
this.initSeqList();
}
if (this.isFull()) {
this.elem = Arrays.copyOf(this.elem, 2 * this.elem.length);
}
this.elem[this.size] = value;
this.size++;
}
🍇2.4指定位置插入数据
这个与顺序插入很相似,只不过多了一个条件:在指定位置插入,那怎么解决这个问题?你不能直接接访问这个位置然后修改这个位置元素的值,这是修改,不是插入。由于顺序表的“指针”是指向最后一个元素的后一个空间,所以我们需要先将指定位置的数据以及后面的数据先全部往后移动一个位置,然后再将指定位置的值改为目标数据。假设需要在顺序表下标为3 的位置插入一个数字22 。
public void add(int pos, int data) {
if (this.elem == null) {
this.initSeqList();
}
if (pos < 0 || pos > this.size) {
System.out.println("插入位置非法!");
return;
}
if (isFull()) {
this.elem = Arrays.copyOf(this.elem, 2 * this.elem.length);
}
for (int i = size - 1; i >= pos; i--) {
this.elem[i+1] = this.elem[i];
}
this.elem[pos] = data;
this.size++;
}
🍇2.5打印顺序表
遍历大法就能搞定!
public void display() {
for (int i = 0; i < this.size; i++) {
System.out.print(this.elem[i] + " ");
}
System.out.println();
}
🍇2.6判断顺序表中是否包含某数据
遍历顺序表,找到就返回。
🍅2.6.1判定是否包含某个元素
public boolean contains(int toFind) {
for (int j : this.elem) {
if (j == toFind) {
return true;
}
}
return false;
}
🍅2.6.2查找某个元素对应的位置
public int search(int toFind) {
for (int i = 0; i < this.size; i++) {
if (this.elem[i] == toFind) {
return i;
}
}
return -1;
}
🍇2.7指定位置的值
先判断位置是否合理,再直接访问返回。
🍅2.7.1获取指定位置元素
public int getPos(int pos) {
if (pos < 0 || pos >= this.size) {
System.out.println("位置非法!");
return -1;
}
return this.elem[pos];
}
🍅2.7.2修改指定位置元素
public void setPos(int pos, int value) {
if (pos < 0 || pos >= this.size) {
System.out.println("修改位置非法!");
return;
}
this.elem[pos] = value;
}
🍇2.8删除第一次出现的指定数据
首先判断顺序表是否为空,为空直接返回。如果顺序表不为空再遍历找到元素并删除。
public void remove(int toRemove) {
if (isEmpty()) {
System.out.println("顺序表为空!");
return;
}
for (int i = 0; i < this.size; i++) {
if (this.elem[i] == toRemove) {
for (int j = i; j < this.size - 1; j++) {
this.elem[j] = this.elem[j+1];
}
this.size--;
return;
}
}
System.out.println("未找到!");
}
🍇2.9获取有效元素个数
访问size 的值即可。
public int size() {
return this.size;
}
🍇2.10删除整个顺序表
这个要注意一件事,如果顺序表的数据是引用类型的,需要先将所有元素的值全部置位null 再将size 置为0 。当然,如果是基本数据,直接将size 置为0 即可。
public void clear() {
this.size = 0;
}
🍓3.源代码
🍇3.1动态顺序表实现代码
import java.util.Arrays;
class StaticSeq {
public int Maxsize;
int[] elem = new int[Maxsize];
public int size;
public StaticSeq(int capacity) {
this.Maxsize = capacity;
}
}
class DynamicSeq {
public int[] elem;
public int size;
}
public class SeqList {
public int[] elem;
public int size;
public SeqList() {
this.initSeqList();
}
public void initSeqList(){
this.elem = new int[10];
}
public void display() {
for (int i = 0; i < this.size; i++) {
System.out.print(this.elem[i] + " ");
}
System.out.println();
}
public void add(int pos, int data) {
if (this.elem == null) {
this.initSeqList();
}
if (pos < 0 || pos > this.size) {
System.out.println("插入位置非法!");
return;
}
if (isFull()) {
this.elem = Arrays.copyOf(this.elem, 2 * this.elem.length);
}
for (int i = size - 1; i >= pos; i--) {
this.elem[i+1] = this.elem[i];
}
this.elem[pos] = data;
this.size++;
}
public boolean isFull() {
return this.size == this.elem.length;
}
public boolean isEmpty() {
return this.size == 0;
}
public boolean contains(int toFind) {
for (int j : this.elem) {
if (j == toFind) {
return true;
}
}
return false;
}
public int search(int toFind) {
for (int i = 0; i < this.size; i++) {
if (this.elem[i] == toFind) {
return i;
}
}
return -1;
}
public void seqAdd(int value) {
if (this.elem == null) {
this.initSeqList();
}
if (this.isFull()) {
this.elem = Arrays.copyOf(this.elem, 2 * this.elem.length);
}
this.elem[this.size] = value;
this.size++;
}
public int getPos(int pos) {
if (pos < 0 || pos >= this.size) {
System.out.println("位置非法!");
return -1;
}
return this.elem[pos];
}
public void setPos(int pos, int value) {
if (pos < 0 || pos >= this.size) {
System.out.println("修改位置非法!");
return;
}
this.elem[pos] = value;
}
public void remove(int toRemove) {
if (isEmpty()) {
System.out.println("顺序表为空!");
return;
}
for (int i = 0; i < this.size; i++) {
if (this.elem[i] == toRemove) {
for (int j = i; j < this.size - 1; j++) {
this.elem[j] = this.elem[j+1];
}
this.size--;
return;
}
}
System.out.println("未找到!");
}
public int size() {
return this.size;
}
public void clear() {
this.size = 0;
}
}
🍇3.2测试专用代码
public class Test {
public static void main(String[] args) {
SeqList sl = new SeqList();
for (int i = 0; i < 12; i++) {
sl.seqAdd(i+1);
}
sl.display();
System.out.println("------------");
sl.add(2,99);
sl.add(13,99);
sl.add(0,99);
sl.display();
System.out.println("------------");
System.out.println(sl.contains(78));
System.out.println(sl.contains(2));
System.out.println("------------");
System.out.println(sl.search(78));
System.out.println(sl.search(5));
System.out.println("------------");
System.out.println(sl.getPos(98));
System.out.println(sl.getPos(5));
System.out.println("------------");
sl.setPos(45, 88);
sl.display();
sl.setPos(1, 88);
sl.display();
System.out.println("------------");
sl.remove(99);
sl.display();
sl.remove(99);
sl.display();
sl.remove(99);
sl.display();
sl.remove(122);
sl.display();
System.out.println("------------");
System.out.println(sl.size);
System.out.println("------------");
sl.clear();
sl.display();
}
}
🍇3.3项目文件
博主的gitee,平常博主写的程序代码都在里面。 博主的github,平常博主写的程序代码都在里面。
觉得文章写得不错的老铁们,点赞评论关注走一波!谢谢啦!
|