Python實戰操作:解題之被圍繞的區域
題目
給定一個二維的矩陣,包含 ‘X’ 和 ‘O’( 字母 O)。
找到所有被 ‘X’ 圍繞的區域,並將這些區域裡所有的 ‘O’ 用 ‘X’ 填充。
示例:
X X X X
X O O X
X X O X
X O X X
執行你的函式後,矩陣變為:
X X X X
X X X X
X X X X
X O X X
解釋:
被圍繞的區間不會存在於邊界上,換句話說,任何邊界上的 ‘O’ 都不會被填充為 ‘X’。 任何不在邊界上,或不與邊界上的 ‘O’ 相連的 ‘O’ 最終都會被填充為 ‘X’。如果兩個元素在水平或垂直方向相鄰,則稱它們是“相連”的。
解題思路
思路:DFS、BFS
首先先看題目,給定的二維矩陣中,包含 ‘X’ 和 ‘O’( 字母 O)。再看解釋部分,任何邊界上的 ‘O’ 不會被填充為 ‘X’,那麼現在也就是說,其實有三個部分:
- 可形成包圍的 ‘X’;
- 被 ‘X’ 包圍的 ‘O’;
- 未被 ‘X’ 包圍的 ‘O’。
現在題目的要求是將被包圍的 ‘O’,轉變為 ‘X’。前面也說了,未被包圍的 ‘O’,是處於邊界上的,同時側面說明與邊界 ‘O’ 相連的 ‘O’ 又不會被 ‘X’ 填充。
那麼我們可以根據這個性質,我們考慮從邊界開始擴散,先去標記不能被填充的 ‘O’,也就是與邊界 ‘O’ 相連的部分,那麼剩下的 ‘O’ 則會被包圍轉變為 ‘X’。具體的做法如下:
- 以邊界的 ‘O’ 為起點,標記與之相連或者間接相連的字母 ‘O’;
- 當標記完上面部分的 ‘O’,此時開始遍歷矩陣,去判斷每個字母(注意:標記的為不會被替換的部分):如果該字母是被標記的部分,那麼將其重新轉換為 ‘O’;如果該字母是未被標記的部門,那麼將其轉換為 ‘X’。
注意:題目要求的是原地修改,那麼我們標記的時候,將與邊界 ‘O’ 相連的部分(包括邊界 ‘O’),標記為 ‘M’。
我們現在使用深度優先搜尋(DFS)和廣度優先搜尋(BFS)的方法來解決這個問題。具體的程式碼如下。
程式碼實現
#深度優先搜尋(DFS)
class Solution:
def solve(self, board: List[List[str]]) -> None:
""
"
Do not return anything, modify board in-place instead.
"
""
if not board or
len(board)==
0:
return
def dfs(board, i, j):
# 不處於矩陣內,或者如果已經標記的話,直接跳過
if not (
0<=i<m) or not (
0<=j<n) or board[i][j] !=
'O':
return
# 當確認為未標記的,並與邊界
'O' 直接間接相連的
'O' 時,標記為
'M'
board[i][j] =
'M'
# 向四個方位擴散
# 上下左右
dfs(board, i
-1, j)
dfs(board, i+
1, j)
dfs(board, i, j
-1)
dfs(board, i, j+
1)
m =
len(board)
n =
len(board[
0])
# 從邊界的
'O' 開始搜尋
for i in
range(m):
for j in
range(n):
# 先確認 i,j 是否處於邊界
is_frontier = (i ==
0 or j ==
0 or i == m
-1 or j == n
-1)
if is_frontier and board[i][j] ==
'O':
dfs(board, i, j)
# 遍歷二維陣列,將被標記為
'M' 的重新轉換為
'O',未標記的,則轉換為
'X'
for i in
range(m):
for j in
range(n):
if board[i][j] ==
'O':
board[i][j] =
'X'
if board[i][j] ==
'M':
board[i][j] =
'O'
# 廣度優先搜尋(BFS)
class Solution:
def solve(self, board: List[List[str]]) -> None:
""
"
Do not return anything, modify board in-place instead.
"
""
if not board or
len(board) ==
0:
return
m =
len(board)
n =
len(board[
0])
from collections
import deque
queue = deque()
# bfs 不同於 dfs 順著滿足條件的方向繼續搜尋
# bfs 滿足條件的都要入佇列
# 先將邊界的
'O' 入隊
for i in
range(m):
# 這裡是二維矩陣的最左列與最右列
if board[i][
0] ==
'O':
queue.
append((i,
0))
if board[i][n
-1] ==
'O':
queue.
append((i, n
-1))
for j in
range(n):
# 這裡是二維矩陣的第一行和最後一行
if board[
0][j] ==
'O':
queue.
append((
0, j))
if board[m
-1][j] ==
'O':
queue.
append((m
-1, j))
# 現在開始出隊,目前裡面的都是邊界的
'O'
# 出隊的同時,進行標記,同時查詢與邊界
'O' 相連的部分入隊
while queue:
x, y = queue.popleft()
board[x][y] =
'M'
# 查詢相連部分
for nx, ny in [(x
-1, y), (x+
1, y), (x, y
-1), (x, y+
1)]:
if
0 <= nx < m and
0 <= ny < n and board[nx][ny] ==
'O':
queue.
append((nx, ny))
# 同樣的,標記完後,遍歷整個二維矩陣,進行轉換
for i in
range(m):
for j in
range(n):
if board[i][j] ==
'O':
board[i][j] =
'X'
if board[i][j] ==
'M':
board[i][j] =
'O'
結果大家可以自己動手實現一下哦!
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/69923331/viewspace-2711544/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- LeetCode-130-被圍繞的區域LeetCode
- LeetCode刷題記112-130. 被圍繞的區域LeetCode
- leetcode 130被圍繞的區域 回溯演算法LeetCode演算法
- Python實戰之Oracle資料庫操作PythonOracle資料庫
- 產品管理圍繞的五個核心問題
- 區域網安全實戰教程
- 實戰前端跨域問題前端跨域
- canvas 圍繞中心旋轉Canvas
- 圍繞央行系統升級所產生的常見問題
- python實戰之爬蟲面試必備題目Python爬蟲面試
- 域滲透之ATT&CK實戰系列——紅隊實戰(一)
- OpenFaaS實戰之四:模板操作(template)
- 區域性範圍掃描的靈活應用
- App 多區域皮膚(主題)的實現APP
- jQuery實戰之 attr() 和 prop() 的區別jQuery
- Prometheus PromQL 講解與實戰操作PrometheusMQ
- 移動端開發——關於區域性區域滾動總結 | 實戰系列
- canvas 圖形圍繞中心旋轉Canvas
- Python閉包區域性變數問題Python變數
- 圍繞低程式碼應用程式開發存在的三個誤解
- Golang的值型別和引用型別的範圍、儲存區域、區別Golang型別
- 手機開發實戰20——GPRS區域定義
- Python區域性函式及用法詳解Python函式
- matlab 繪製置信範圍_fill(繪製其區間形成的區域)Matlab
- 圍繞DOM元素節點的增刪改查
- Flutter 入門與實戰(五十四):Provider 之監聽狀態的區域性變化FlutterIDE
- 區域網內獲取周圍裝置的ip和埠
- svg矩形圍繞自身中心旋轉效果SVG
- 挑戰Python題解-019Python
- 上傳靶機實戰之upload-labs解題
- 挑戰系統 / 進入區域挑戰怪物
- Python剪裁影像中的指定區域Python
- Python爬蟲實戰之bilibiliPython爬蟲
- python opencv如何實現目標區域裁剪功能PythonOpenCV
- 幾個MySQL在Python中操作示例,MySQL利用於Python的實戰!MySqlPython
- 詞!自然語言處理之詞全解和Python實戰!自然語言處理Python
- 百度地圖獲取多行政區域圍欄地圖
- 專案實戰之跨域處理~一文搞定所有跨域需求跨域