【图的深度优先遍历搜索】Seeding
生活随笔
收集整理的這篇文章主要介紹了
【图的深度优先遍历搜索】Seeding
小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
【問題描述】
湯姆有一塊好田,它是一個n×m方格的矩形,某些方格里面有一些大石頭。湯姆有一臺播種機。開始的時候,機器位于田地的左上角。機器播種完一個方格后,湯姆就把它開到一個相鄰的方格里,繼續播種。為了保護機器,湯姆不會把機器開到有石頭的方格里面,也不會開到剛剛播種過的方格里面。 湯姆希望在沒有石頭的方格里面都能播種,這可能嗎?
輸入
每個測試例的第一行包含兩個整數n和m,表示田地的大小。 接下來n行描述田地,每行包含m個字符:‘S’表示方格里面有石頭,‘.’表示方格里面沒有石頭。
輸出
對每個測試例,如果湯姆播種完畢,輸出“YES”,否則輸出“NO”。
| 輸入樣例 | 4?4 .S.. .S.. .... .... | 4?4 .... ...S .... ...S | 0?0 |
| 輸出樣例 | YES | NO |
【算法分析】
(1)數據結構
//播種問題的數據結構 #define NUM 10 char field[NUM][NUM]; //田地 int n, m; //田地的大小 int visited; //訪問田地中方格的數量 int flag; //全部播種完畢的標志 int count; //遞歸的次數計數(2)判斷播種機能否播種完全部農田的深度優先搜索算法
void dfs(int x,int y) //形參是方格的坐標(x,y) {if ( x<0 || y<0 || x>=n || y>=m) return;if (field[x][y]=='S') return;if (flag) return; //已經全部播種完畢//遞歸的次數計數,限制不要超過1500次count ++;if (count>1500) return;//將方格里面置為石頭,避免重復搜索field[x][y] = 'S';visited ++; //訪問的方格計數if (visited == n*m) //全部方格訪問完畢{flag = 1;return;}//分別向4個鄰近方格播種dfs(x+1, y);dfs(x-1, y);dfs(x, y+1);dfs(x, y-1);visited --; //恢復遞歸現場field[x][y] = '.'; }①是不是要等到所有遞歸都結束時,才知道播種機能否播種所有的可播種方格呢?測試表明,當遞歸的次數達到1500時,標志flag基本上已經明確。
②限制visited<1500(經在線測試摸索),就可以限制很多不必要的搜索,在線測試時間降到0ms,否則是120ms。
//變量visited表示訪問田地中方格的數量,初始時visited是田地中有石頭方格的數量visited=0; for(int i=0;i<n;i++)for(int j=0;j<m;j++)if(field[i][j]=='S') visited++;【代碼的完整實現】
#include<iostream>#define NUM 10 char field[NUM][NUM]; //田地 int n,m; //田地大小 int visited; //訪問田地中方格的數量 int flag; //全部播種完畢的標志 int count; //遞歸次數統計 void dfs(int x,int y) //坐標 {if ( x<0 || y<0 || x>=n || y>=m) return;if (field[x][y]=='S') return;//已經全部播種完畢//遞歸的次數計數,限制不要超過1500次if (flag) return;count ++;if (count>1500) return;//將方格里面置為石頭,避免重復搜索field[x][y] = 'S';visited ++; //訪問方格計數 if (visited == n*m) //全部方格訪問完畢 {flag = 1;return;}//分別向4個鄰近方格播種dfs(x+1,y);dfs(x-1,y);dfs(x,y+1);dfs(x,y-1);visited --; //恢復遞歸狀態 field[x][y] = '.'; }int main() {while (scanf("%d%d",&n,&m) && n && m) //輸入田地大小 {int i, j;for (i=0; i<n; i++)scanf("%s", field[i]); //輸入方格里面是否為石頭 visited = 0;for (i = 0; i<n ; i++)for (j = 0; j<m; j++)if (field[i][j]=='S') visited ++; //有石頭則訪問+1 flag = 0; //初始化 count = 0;dfs(0,0); //從(0,0)處開始遞歸處理 if (flag) printf("YES\n");else printf("NO\n");} return 0; }總結
以上是生活随笔為你收集整理的【图的深度优先遍历搜索】Seeding的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: React-router4 第五篇 Pr
- 下一篇: 微型计算机k80,AEG 3AEF151