一、稀疏数组和队列
1.稀疏sparsearray数组
1.1 定义
1.2 思路
-
二维数组 转 稀疏数组的思路
- 遍历 原始的二维数组,得到有效数据的个数 sum
- 根据sum 就可以创建 稀疏数组 sparseArr int[sum + 1] [3]
- 将二维数组的有效数据数据存入到 稀疏数组
-
稀疏数组转原始的二维数组的思路
-
先读取稀疏数组的第一行,根据第一行的数据,创建原始的二维数组,比如上面的
chessArr2 = int [11][11]
-
在读取稀疏数组后几行的数据,并赋给 原始的二维数组 即可.
-
1.3 代码
public class SparseArray {
public static void main(String[] args) {
//创建一个原始的二维数组 11*11
//0:表示没有棋子,1:表示黑子 2:表示蓝子
int chessArr1[][] = new int[11][11];
chessArr1[1][2] = 1;
chessArr1[2][3] = 2;
chessArr1[4][5] = 2;
//输出原始的二维数组
System.out.println("原始的二维数组");
for (int[] row :
chessArr1) {
for (int data :
row) {
System.out.print("\t" + data);
}
System.out.println();
}
//将二维数组转稀疏数组
//1.先遍历二维数组 得到非0数据的个数
int sum = 0;
for (int i = 0; i < chessArr1.length; i++) {
for(int j = 0 ; j < chessArr1[i].length; j++){
if(chessArr1[i][j] != 0){
sum++;
}
}
}
System.out.println("sum = " + sum);
//2.创建对应的稀疏数组
int sparseArr[][] = new int[sum + 1][3];
//给稀疏数组赋值
sparseArr[0][0] = chessArr1.length;
sparseArr[0][1] = chessArr1[0].length;
sparseArr[0][2] = sum;
//遍历二维数组,将非0的数据存放到稀疏数组里面
int count = 0;//count用于记录是第几个非0数组
for (int i = 0; i < chessArr1.length; i++) {
for(int j = 0 ; j < chessArr1[i].length; j++){
if(chessArr1[i][j] != 0){
count++;
sparseArr[count][0] = i;
sparseArr[count][1] = j;
sparseArr[count][2] = chessArr1[i][j];
}
}
}
//输出稀疏数组的形式
System.out.println("得到的稀疏数组为:");
for (int[] row :
sparseArr) {
for (int data :
row) {
System.out.print("\t" + data);
}
System.out.println();
}
//将稀疏数组恢复成原始的二维数组
/*
1.先读取稀疏数组的第一行,根据第一行的数据,创建原始的二维数组。
2.再读取稀疏数组后几行的数据,并赋值给原始的二维数组即可
*/
System.out.println("还原完的数据如下");
int[][] newArray = new int[sparseArr[0][0]][sparseArr[0][1]];
for(int i = 1;i < sparseArr.length;i++){
newArray[sparseArr[i][0]][sparseArr[i][1]] = sparseArr[i][2];
}
for (int[] row :
newArray) {
for (int data :
row) {
System.out.print("\t" + data);
}
System.out.println();
}
}
}
2.队列
2.1 定义
-
队列是一个有序列表,可以用数组或是链表来实现。
-
遵循先入先出的原则。即:先存入队列的数据,要先取出。后存入的要后取出
-
示意图:(使用数组模拟队列示意图)
2.2 思路
思路如下:
-
front 变量的含义做一个调整: front 就指向队列的第一个元素, 也就是说 arr[front] 就是队列的第一个元素front 的初始值 = 0
-
rear 变量的含义做一个调整:rear 指向队列的最后一个元素的后一个位置. 因为希望空出一个空间做为约定.rear 的初始值 = 0
-
当队列满时,条件是 (rear + 1) % maxSize == front 【满】
-
对队列为空的条件, rear == front 空
-
当我们这样分析, 队列中有效的数据的个数 (rear + maxSize - front) % maxSize // rear = 1 front = 0
-
我们就可以在原来的队列上修改得到,一个环形队列
2.3 循环队列代码
2.3.1 说明
数组模拟环形队列
对前面的数组模拟队列的优化,充分利用数组. 因此将数组看做是一个环形的。(通过取模的方式来实现即可)
分析说明:
尾索引的下一个为头索引时表示队列满,即将队列容量空出一个作为约定,这个在做判断队列满的时候需要注意 (rear + 1) % maxSize == front 满]
rear == front [空]
2.3.2 代码
package 队列02;
import java.util.Scanner;
public class ArrayQueueDemo {
public static void main(String[] args) {
//测试
System.out.println("测试数组模拟环形队列的案例");
ArrayQueue queue = new ArrayQueue(4);//这里设置4,其队列的有效数据最大值是3
char key = ' ';//接收用户输入
Scanner scanner = new Scanner(System.in);
boolean loop = true;
//输出一个菜单
while (loop){
System.out.println("s(show):显示队列");
System.out.println("e(exit):退出程序");
System.out.println("a(add):添加数据到队列");
System.out.println("g(get):从队列取出数据");
System.out.println("h(head):查看队列头的数据");
key = scanner.next().charAt(0);//接受一个字符
switch (key){
case 's':
queue.showQueue();
break;
case 'a':
System.out.println("输出一个数");
int value = scanner.nextInt();
queue.addData(value);
break;
case 'g':
try {
int res = queue.getQueue();
System.out.printf("取出的数据是%d\n",res);
} catch (Exception e) {
System.out.println(e.getMessage());
}
break;
case 'h':
try {
int res = queue.headQueue();
System.out.printf("队列头的数据是%d\n",res);
} catch (Exception e) {
System.out.println(e.getMessage());
}
break;
case 'e'://退出
scanner.close();
loop = false;
break;
}
}
System.out.println("程序退出");
}
}
//使用数组模拟队列-编写一个ArrayQueue类
//目前数据使用一次就不能使用了,没有达到服用的效果
class ArrayQueue {
private int front;
private int rear;
private int[] arr;
private int maxSize;
public ArrayQueue(int maxSize){
this.maxSize = maxSize;
arr = new int[maxSize];
front = 0;
rear = 0;
}
public boolean isEmpty(){
return front == rear;
}
public boolean isFull(){
return (rear + 1) % maxSize == front;
}
public void addData(int data){
if(isFull()){
System.out.println("队列满了");
}
arr[rear] = data;
//将rear后移,这里必须考虑取模
rear = (rear + 1 ) % maxSize;
}
public int getQueue(){
if(isEmpty()){
throw new RuntimeException("没有数据");
}
//这里需要分析出front是指向队列的第一个元素
//1.先把front的对应的值保存到一个临时的变量
//2.将front后移,考虑取模
//3.将临时保存的变量返回
int value = arr[front];
front = (front + 1) % maxSize;
return value;
}
//显示队列的所有数据
public void showQueue(){
//遍历
if(isEmpty()){
System.out.println("队列为空,没有数据");
return;
}
//思路:从front开始遍历,遍历多少个元素
for (int i = front; i < front + size(); i++) {
System.out.printf("arr[%d] = %d\n",i % maxSize,arr[i % maxSize]);
}
}
//求出当前队列有效数据的个数
public int size(){
return (rear + maxSize - front) % maxSize;
}
//显示队列的头数据,注意不是取出数据
public int headQueue(){
//判断
if(isEmpty()){
throw new RuntimeException("队列为空,没有数据");
}
return arr[front];
}
}
据的个数
public int size(){
return (rear + maxSize - front) % maxSize;
}
//显示队列的头数据,注意不是取出数据
public int headQueue(){
//判断
if(isEmpty()){
throw new RuntimeException("队列为空,没有数据");
}
return arr[front];
}
}