Java 超詳細(xì)講解數(shù)據(jù)結(jié)構(gòu)的應(yīng)用
一.bfs
bfs(廣度優(yōu)先搜索),類似二叉樹的層序遍歷,利用隊(duì)列完成。一般用于求最短路。
圖的最短路問(wèn)題:
給定一個(gè)無(wú)向圖,每條邊的長(zhǎng)度都是1。求1號(hào)點(diǎn)到x號(hào)點(diǎn)的最短距離。 頂點(diǎn)數(shù)n ?邊數(shù)為m
q次詢問(wèn) ?輸入x 輸出1到x的最短距離。 若1號(hào)點(diǎn)到x不連通,則輸出-1
二.雙端隊(duì)列
雙端隊(duì)列的應(yīng)用(區(qū)間翻轉(zhuǎn)):
對(duì)于長(zhǎng)度為n的數(shù)組,給定一個(gè)長(zhǎng)度為m的區(qū)間,區(qū)間初始位置為a[1]到a[m]。
3種操作:
- 區(qū)間右移(最右端不會(huì)超過(guò)a[n])
- 區(qū)間左移(最左端不會(huì)超過(guò)a[n])
- 區(qū)間內(nèi)所有數(shù)翻轉(zhuǎn)。
q次操作后請(qǐng)你還原數(shù)組。
三.算法題
1.kotori和迷宮
難度??
知識(shí)點(diǎn):bfs
首先找到k字母,然后從k字母位置開(kāi)始bfs。bfs過(guò)程中即可得到k到每個(gè)e的最短路程。(要注意走過(guò)的e不可繼續(xù)往下走)
題目描述:
kotori在一個(gè)n*m迷宮里,迷宮的最外層被巖漿淹沒(méi),無(wú)法涉足,迷宮內(nèi)有k個(gè)出口。kotori只能上下左右四個(gè)方向移動(dòng)。她想知道有多少出口是她能到達(dá)的,最近的出口離她有多遠(yuǎn)?
輸入描述:
第一行為兩個(gè)整數(shù)n和m,代表迷宮的行和列數(shù) (1≤n,m≤30)
后面緊跟著n行長(zhǎng)度為m的字符串來(lái)描述迷宮。'k'代表kotori開(kāi)始的位置,'.'代表道路,'*'代表墻壁,'e'代表出口。保證輸入合法。
輸出描述:
若有出口可以抵達(dá),則輸出2個(gè)整數(shù),第一個(gè)代表kotori可選擇的出口的數(shù)量,第二個(gè)代表kotori到最近的出口的步數(shù)。(注意,kotori到達(dá)出口一定會(huì)離開(kāi)迷宮)
若沒(méi)有出口可以抵達(dá),則輸出-1。
示例1
輸入
6 8
e.*.*e.*
.**.*.*e
..*k**..
***.*.e*
.**.*.**
*......e
輸出
2 7
說(shuō)明
可供選擇坐標(biāo)為[4,7]和[6,8],到kotori的距離分別是8和7步。
import java.util.*;
import java.io.*;
public class Main{
public static void main(String[] args) throws IOException{
BufferedReader bf = new BufferedReader(new InputStreamReader(System.in));
String[] s1 = bf.readLine().split(" ");
int n = Integer.parseInt(s1[0]);
int m = Integer.parseInt(s1[1]);
//建立地圖、標(biāo)記圖
char[][] maze = new char[n][m];
boolean[][] visited = new boolean[n][m];
//紀(jì)錄步數(shù)
int[][] dis = new int[n][m];
//紀(jì)錄初始的坐標(biāo)
int ki = 0, kj = 0;
for(int i = 0; i < n; i++){
String s = bf.readLine();
for(int j = 0; j < m; j++){
dis[i][j] = Integer.MAX_VALUE;
char c = s.charAt(j);
maze[i][j] = c;
if(c == 'k'){
ki = i;
kj = j;
}
}
}
int count = 0, min = Integer.MAX_VALUE;
Queue<Integer> queue = new ArrayDeque<>();
//二維數(shù)組的性質(zhì),保存了坐標(biāo),并且節(jié)省了空間
queue.add(ki * m + kj);
visited[ki][kj] = true;
dis[ki][kj]= 0;
while(!queue.isEmpty()){
int temp = queue.poll();
int tempi = temp / m, tempj = temp % m;
//支持八個(gè)方向的移動(dòng)或者不移動(dòng)(但是因?yàn)镸ath.abs(i - j) == 1限定了絕對(duì)值為1,所以變成了四個(gè)方向)
for(int i = -1; i <= 1; i++){
for(int j = -1; j <= 1; j++){
if(Math.abs(i - j) == 1 && tempi + i >= 0 && tempi + i < n && tempj + j >= 0 && tempj + j < m && !visited[tempi + i][tempj + j]){
if(maze[tempi + i][tempj + j] == '.'){
visited[tempi + i][tempj + j] = true;
dis[tempi + i][tempj + j] = dis[tempi][tempj] + 1;
queue.add((tempi + i) * m + (tempj + j));
}
if(maze[tempi + i][tempj + j] == 'e'){
visited[tempi + i][tempj + j] = true;
dis[tempi + i][tempj + j] = dis[tempi][tempj] + 1;
min = Math.min(min, dis[tempi][tempj] + 1);
count++;
}
}
}
}
}
if(count == 0) System.out.print(-1);
else System.out.print(count + " " + min);
}
}
思考:隊(duì)列是怎么實(shí)現(xiàn)bfs的?
1.起始點(diǎn)入隊(duì)-->2.將起始點(diǎn)四個(gè)方向的可達(dá)點(diǎn)入隊(duì)-->3.起始點(diǎn)出隊(duì)。以此循序依次訪問(wèn)隊(duì)列中的元素。
2.小紅找紅點(diǎn)
難度???
知識(shí)點(diǎn):bfs,多源最短路
多源最短路的求法:在bfs開(kāi)始之前將所有點(diǎn)都扔進(jìn)隊(duì)列,然后開(kāi)始bfs即可。
題目描述:
小紅拿到了一張無(wú)向圖,有 n?個(gè)頂點(diǎn)和?m?條邊。每條邊的長(zhǎng)度為 1 。?
小紅給一些頂點(diǎn)染成了紅色。她想知道,對(duì)于每個(gè)頂點(diǎn),到附近最近的紅色點(diǎn)的距離為多少?
輸入描述:
第一行輸出兩個(gè)正整數(shù) n 和 m ,用空格隔開(kāi)。分別代表頂點(diǎn)數(shù)和邊數(shù)。
第二行輸入一個(gè)長(zhǎng)度為 n 的字符串,代表每個(gè)頂點(diǎn)的染色情況。第?i 個(gè)字符為 'R' 代表被染成紅色,為 'W' 代表未被染色。
接下來(lái)的?m 行,每行兩個(gè)正整數(shù) x 和?y ,代表?x 和?y 有一條無(wú)向邊相連。
不保證圖是整體連通的。不保證沒(méi)有重邊和自環(huán)。
1<=n,m<=10^5
輸出描述:
輸出一行 n 個(gè)整數(shù),代表從?1 到 n 每個(gè)頂點(diǎn)到最近的紅色頂點(diǎn)的距離。若對(duì)于某點(diǎn)而言無(wú)論如何都走不到紅色頂點(diǎn),則輸出 -1 。
示例1:
輸入
5 5
RWWRW
1 2
3 3
1 2
2 5
1 4
輸出
0 1 -1 0 2
說(shuō)明

樣例的圖如上所示。
import java.util.*;
import java.io.*;
public class Main{
static ArrayList<Integer>[] g;
static String[] strings;
static int[] visited;
static int[] dis;
public static void main(String[] args) throws Exception {
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
String[] firstLine = br.readLine().split(" ");
int n = Integer.parseInt(firstLine[0]);
int m = Integer.parseInt(firstLine[1]);
g = new ArrayList[n+1];
visited = new int[n+1];
dis= new int[n+1];
for (int i=1;i<n+1;i++) {
g[i] = new ArrayList<Integer>();
}
//一個(gè)字符一個(gè)字符的讀取
strings = br.readLine().split("");
for (int i=0;i<m;i++) {
//描繪雙向圖
String[] temp = br.readLine().split(" ");
int x = Integer.parseInt(temp[0]);
int y = Integer.parseInt(temp[1]);
g[x].add(y);
g[y].add(x);
}
//g[x]代表當(dāng)前點(diǎn) g[x].get(i)代表所連的線
Queue<Integer> queue = new ArrayDeque<>();
for(int i=1;i<=n;i++){
if(strings[i-1].equals("R")){
queue.add(i);
visited[i]=1;
}
}
while(!queue.isEmpty()){
int temp=queue.remove();
for(int i=0;i<g[temp].size();i++){
if(visited[g[temp].get(i)]==0){
visited[g[temp].get(i)]=1;
dis[g[temp].get(i)]=dis[temp]+1;
queue.add(g[temp].get(i));
}
}
}
for(int i=1;i<=n;i++){
if(visited[i]==0)System.out.print("-1 ");
else System.out.print(dis[i]+" ");
}
}
}
對(duì)照上一章的案例:小紅點(diǎn)點(diǎn)點(diǎn)結(jié)合理解。? 分別使用的dfs和bfs。
本題思想:先將紅色的所有點(diǎn)都入隊(duì)列,然后bfs。
這是一種逆向思維:不是所謂的從編號(hào)開(kāi)始,并且所有走過(guò)的都不能在走了。
3.小紅玩數(shù)組?
難度????
知識(shí)點(diǎn):雙端隊(duì)列
用一個(gè)雙端隊(duì)列來(lái)模擬過(guò)程,用一個(gè)變量來(lái)標(biāo)記雙端隊(duì)列是否翻轉(zhuǎn)過(guò)。


示例1:
輸入
6 4
1 5 4 6 2 8
5
21323
輸出
4 6 2 1 5 8
import java.io.*;
import java.util.*;
public class Main{
static Deque<Integer> workQueue;
public static void main(String[] args)throws IOException{
BufferedReader br=new BufferedReader(new InputStreamReader(System.in));
PrintWriter pw=new PrintWriter(System.out);
String[] firstLine=br.readLine().split(" ");
int total=Integer.parseInt(firstLine[0]);
int size=Integer.parseInt(firstLine[1]);
int[] arr=new int[total];
String[] secondLine=br.readLine().split(" ");
for(int i=0;i<total;i++){
arr[i]=Integer.parseInt(secondLine[i]);
}
int L=0;
int R=size-1;
workQueue=new LinkedList<>();
for(int i=0;i<size;i++){
workQueue.offerLast(arr[i]);
}
int times=Integer.parseInt(br.readLine());
String tries=br.readLine();
int is=0;//0代表沒(méi)有翻轉(zhuǎn)!
for(int i=0;i<times;i++){
if(tries.charAt(i)=='1'){
if(R==arr.length-1)
continue;
R++;
if(is==0){
workQueue.offerLast(arr[R]);
int tmp=workQueue.pollFirst();
arr[L]=tmp;
}else{
workQueue.offerFirst(arr[R]);
int tmp=workQueue.pollLast();
arr[L]=tmp;
}
L++;
}else if(tries.charAt(i)=='2'){
if(L==0)
continue;
L--;
if(is==0){
workQueue.offerFirst(arr[L]);
arr[R]=workQueue.pollLast();
}else{
workQueue.offerLast(arr[L]);
arr[R]=workQueue.pollFirst();
}
R--;
}else{
is=1-is;
}
}
for(int i=0;i<L;i++){
pw.print(arr[i]+" ");
}
if(is==0){
while(!workQueue.isEmpty()) {
pw.print(workQueue.pollFirst() + " ");
}
}else{
while(!workQueue.isEmpty()) {
pw.print(workQueue.pollLast() + " ");
}
}
for(int i=R+1;i<arr.length;i++){
pw.print(arr[i]+" ");
}
pw.flush();
}
}
到此這篇關(guān)于Java 超詳細(xì)講解數(shù)據(jù)結(jié)構(gòu)的應(yīng)用的文章就介紹到這了,更多相關(guān)Java 數(shù)據(jù)結(jié)構(gòu)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java實(shí)現(xiàn)系統(tǒng)多級(jí)文件夾復(fù)制
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)系統(tǒng)多級(jí)文件夾復(fù)制,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-08-08
Java使用volatile關(guān)鍵字的注意事項(xiàng)
volatile關(guān)鍵字是Java中的一種稍弱的同步機(jī)制,為什么稱之為弱機(jī)制。這篇文章主要介紹了Java使用volatile關(guān)鍵字的注意事項(xiàng),需要的朋友可以參考下2017-02-02
淺析如何在Java應(yīng)用中優(yōu)雅的發(fā)送短信
很多業(yè)務(wù)場(chǎng)景里,我們都需要發(fā)送短信,比如登陸驗(yàn)證碼、告警、營(yíng)銷通知、節(jié)日祝福等等,這篇文章,我們聊聊 Java 應(yīng)用中如何優(yōu)雅的發(fā)送短信,文中有詳細(xì)的代碼示例供大家參考,需要的朋友可以參考下2023-11-11
Spring中基于xml的AOP實(shí)現(xiàn)詳解
這篇文章主要介紹了Spring中基于xml的AOP實(shí)現(xiàn)詳解,基于xml與基于注解的AOP本質(zhì)上是非常相似的,都是需要封裝橫切關(guān)注點(diǎn),封裝到切面中,然后把橫切關(guān)注點(diǎn)封裝為一個(gè)方法,再把該方法設(shè)置為當(dāng)前的一個(gè)通知,再通過(guò)切入點(diǎn)表達(dá)式定位到橫切點(diǎn)就可以了,需要的朋友可以參考下2023-09-09
springboot 默認(rèn)靜態(tài)路徑實(shí)例解析
這篇文章主要介紹了springboot 默認(rèn)靜態(tài)路徑實(shí)例解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-11-11
JAVA讀取文件流,設(shè)置瀏覽器下載或直接預(yù)覽操作
這篇文章主要介紹了JAVA讀取文件流,設(shè)置瀏覽器下載或直接預(yù)覽操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-10-10
新建Maven工程出現(xiàn)Process?Terminated的問(wèn)題解決
當(dāng)Maven出現(xiàn)"Process terminated"錯(cuò)誤時(shí),這通常是由于配置文件或路徑錯(cuò)誤導(dǎo)致的,本文主要介紹了新建Maven工程出現(xiàn)Process?Terminated的問(wèn)題解決,感興趣的可以了解一下2024-04-04
Java實(shí)戰(zhàn)之實(shí)現(xiàn)物流配送系統(tǒng)示例詳解
這篇文章主要介紹了一個(gè)java實(shí)戰(zhàn)項(xiàng)目:通過(guò)java、SSM、JSP、mysql和redis實(shí)現(xiàn)一個(gè)物流配送系統(tǒng)。文中的示例代碼非常詳細(xì),需要的朋友可以參考一下2021-12-12

