日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

【图的深度优先遍历搜索】Seeding

發布時間:2024/5/15 编程问答 35 豆豆
生活随笔 收集整理的這篇文章主要介紹了 【图的深度优先遍历搜索】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的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。