用 Python 製作一個迷宮遊戲

2021-03-02 Crossin的編程教室

大家好,歡迎來到 Crossin的編程教室 !

相信大家都玩過迷宮的遊戲,對於簡單的迷宮,我們可以一眼就看出通路,但是對於複雜的迷宮,可能要仔細尋找好久,甚至耗費數天,然後可能還要分別從入口和出口兩頭尋找才能找的到通路,甚至也可能找不到通路。

雖然走迷宮問題對於我們人類來講比較複雜,但對於計算機來說卻是很簡單的問題。為什麼這樣說呢,因為看似複雜實則是有規可循的。

我們可以這麼做,攜帶一根很長的繩子,從入口出發一直走,如果有岔路口就走最左邊的岔口,直到走到死胡同或者找到出路。如果是死胡同則退回上一個岔路口,我們稱之為岔口 A,

這時進入左邊第二個岔口,進入第二個岔口後重複第一個岔口的步驟,直到找到出路或者死胡同退回來。當把該岔路口所有的岔口都走了一遍,還未找到出路就沿著繩子往回走,走到岔口 A 的前一個路口 B,重複上面的步驟。

不知道你有沒有發現,這其實就是一個不斷遞歸的過程,而這正是計算機所擅長的。

上面這種走迷宮的算法就是我們常說的深度優先遍歷算法,與之相對的是廣度優先遍歷算法。有了理論基礎,下面我們就來試著用 程序來實現一個走迷宮的小程序。

先來看看最終的效果視頻。

生成迷宮

生成迷宮有很多種算法,常用的有遞歸回溯法、遞歸分割法和隨機 Prim 算法,我們今天是用的最後一種算法。

該算法的主要步驟如下:

1、迷宮行和列必須為奇數
2、奇數行和奇數列的交叉點為路,其餘點為牆,迷宮四周全是牆
3、選定一個為路的單元格(本例選 [1,1]),然後把它的鄰牆放入列表 wall
4、當列表 wall 裡還有牆時:
    4.1、從列表裡隨機選一面牆,如果這面牆分隔的兩個單元格只有一個單元格被訪問過
        4.1.1、那就從列表裡移除這面牆,同時把牆打通
        4.1.2、將單元格標記為已訪問
        4.1.3、將未訪問的單元格的鄰牆加入列表 wall
    4.2、如果這面牆兩面的單元格都已經被訪問過,那就從列表裡移除這面牆

我們定義一個 Maze 類,用二維數組表示迷宮地圖,其中 1 表示牆壁,0 表示路,然後初始化左上角為入口,右下角為出口,最後定義下方向向量。

class Maze:
    def __init__(self, width, height):
        self.width = width
        self.height = height
        self.map = [[0 if x % 2 == 1 and y % 2 == 1 else 1 for x in range(width)] for y in range(height)]
        self.map[1][0] = 0  # 入口
        self.map[height - 2][width - 1] = 0  # 出口
        self.visited = []
        # right up left down
        self.dx = [1, 0, -1, 0]
        self.dy = [0, -1, 0, 1]

接下來就是生成迷宮的主函數了。

def generate(self):
    start = [1, 1]
    self.visited.append(start)
    wall_list = self.get_neighbor_wall(start)
    while wall_list:
        wall_position = random.choice(wall_list)
        neighbor_road = self.get_neighbor_road(wall_position)
        wall_list.remove(wall_position)
        self.deal_with_not_visited(neighbor_road[0], wall_position, wall_list)
        self.deal_with_not_visited(neighbor_road[1], wall_position, wall_list)

該函數裡面有兩個主要函數 get_neighbor_road(point) 和 deal_with_not_visited(),前者會獲得傳入坐標點 point 的鄰路節點,返回值是一個二維數組,後者 deal_with_not_visited() 函數處理步驟 4.1 的邏輯。

由於 Prim 隨機算法是隨機的從列表中的所有的單元格進行隨機選擇,新加入的單元格和舊加入的單元格被選中的概率是一樣的,因此其分支較多,生成的迷宮較複雜,難度較大,當然看起來也更自然些。生成的迷宮。

[1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1]
[0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1]
[1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1]
[1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1]
[1, 1, 1, 0, 1, 0, 1, 0, 1, 0, 1]
[1, 0, 0, 0, 1, 0, 1, 0, 1, 0, 1]
[1, 0, 1, 0, 1, 0, 1, 1, 1, 1, 1]
[1, 0, 1, 0, 1, 0, 0, 0, 0, 0, 1]
[1, 0, 1, 0, 1, 1, 1, 0, 1, 0, 1]
[1, 0, 1, 0, 1, 0, 0, 0, 1, 0, 0]
[1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1]

走出迷宮

得到了迷宮的地圖,接下來就按照我們文首的思路來走迷宮即可。主要函數邏輯如下:


def dfs(self, x, y, path, visited=[]):
    # outOfIndex
    if self.is_out_of_index(x, y):
        return False

    # visited or is wall
    if [x, y] in visited or self.get_value([x, y]) == 1:
        return False

    visited.append([x, y])
    path.append([x, y])

    # end...
    if x == self.width - 2 and y == self.height - 2:
        return True

    # recursive
    for i in range(4):
        if 0 < x + self.dx[i] < self.width - 1 and 0 < y + self.dy[i] < self.height - 1 and \
                self.get_value([x + self.dx[i], y + self.dy[i]]) == 0:
            if self.dfs(x + self.dx[i], y + self.dy[i], path, visited):
                return True
            elif not self.is_out_of_index(x, y) and path[-1] != [x, y]:
                path.append([x, y])

很明顯,這就是一個典型的遞歸程序。當該節點坐標越界、該節點被訪問過或者該節點是牆壁的時候,直接返回,因為該節點肯定不是我們要找的路徑的一部分,否則就將該節點加入被訪問過的節點和路徑的集合中。

然後如果該節點是出口則表示程序執行結束,找到了通路。不然就遍歷四個方向向量,將節點的鄰路傳入函數 dfs 繼續以上步驟,直到找到出路或者程序所有節點都遍歷完成。

來看看我們 dfs 得出的路徑結果:

[[0, 1], [1, 1], [2, 1], [3, 1], [4, 1], [5, 1], [6, 1], [7, 1], [8, 1], [9, 1], [9, 1], [8, 1], [7, 1], [6, 1], [5, 1], [5, 2], [5, 3], [6, 3], [7, 3], [8, 3], [9, 3], [9, 4], [9, 5], [9, 5], [9, 4], [9, 3], [8, 3], [7, 3], [7, 4], [7, 5], [7, 5], [7, 4], [7, 3], [6, 3], [5, 3], [4, 3], [3, 3], [2, 3], [1, 3], [1, 3], [2, 3], [3, 3], [3, 4], [3, 5], [2, 5], [1, 5], [1, 6], [1, 7], [1, 8], [1, 9], [1, 9], [1, 8], [1, 7], [1, 6], [1, 5], [2, 5], [3, 5], [3, 6], [3, 7], [3, 8], [3, 9], [3, 9], [3, 8], [3, 7], [3, 6], [3, 5], [3, 4], [3, 3], [4, 3], [5, 3], [5, 4], [5, 5], [5, 6], [5, 7], [6, 7], [7, 7], [8, 7], [9, 7], [9, 8], [9, 9], [10, 9]]

可視化

有了迷宮地圖和通路路徑,剩下的工作就是將這些坐標點渲染出來。今天我們用的可視化庫是 pyxel,這是一個用來寫像素級遊戲的  Python 庫,

當然使用前需要先安裝下這個庫。

Win 用戶直接用 pip install -U pyxel命令安裝即可。

Mac 用戶使用以下命令安裝:

brew install python3 gcc sdl2 sdl2_image gifsicle
pip3 install -U pyxel

先來看個簡單的 Demo。

import pyxel

class App:
    def __init__(self):
        pyxel.init(160, 120)
        self.x = 0
        pyxel.run(self.update, self.draw)

    def update(self):
        self.x = (self.x + 1) % pyxel.width

    def draw(self):
        pyxel.cls(0)
        pyxel.rect(self.x, 0, 8, 8, 9)

App()

類 App 的執行邏輯就是不斷的調用 update 函數和 draw 函數,因此可以在 update 函數中更新物體的坐標,然後在 draw 函數中將圖像畫到屏幕即可。

如此我們就先把迷宮畫出來,然後在渲染 dfs 遍歷動畫。

width, height = 37, 21
my_maze = Maze(width, height)
my_maze.generate()

class App:
    def __init__(self):
        pyxel.init(width * pixel, height * pixel)
        pyxel.run(self.update, self.draw)

    def update(self):
        if pyxel.btn(pyxel.KEY_Q):
            pyxel.quit()

        if pyxel.btn(pyxel.KEY_S):
            self.death = False

    def draw(self):
        # draw maze
        for x in range(height):
            for y in range(width):
                color = road_color if my_maze.map[x][y] is 0 else wall_color
                pyxel.rect(y * pixel, x * pixel, pixel, pixel, color)
        pyxel.rect(0, pixel, pixel, pixel, start_point_color)
        pyxel.rect((width - 1) * pixel, (height - 2) * pixel, pixel, pixel, end_point_color)

App()

看起來還可以,這裡的寬和高我分別用了 37 和 21 個像素格來生成,所以生成的迷宮不是很複雜,如果像素點很多的話就會錯綜複雜了。

接下裡來我們就需要修改 update 函數和 draw 函數來渲染路徑了。為了方便操作,我們在 init 函數中新增幾個屬性。

self.index = 0
self.route = [] # 用於記錄待渲染的路徑
self.step = 1 # 步長,數值越小速度越快,1:每次一格;10:每次 1/10 格
self.color = start_point_color
self.bfs_route = my_maze.bfs_route()

其中 index 和 step 是用來控制渲染速度的,在 draw 函數中 index 每次自增 1,然後再對 step 求餘數得到當前的真實下標 real_index,簡言之就是 index 每增加 step,real_index 才會加一,渲染路徑向前走一步。

def draw(self):
    # draw maze
    for x in range(height):
        for y in range(width):
            color = road_color if my_maze.map[x][y] is 0 else wall_color
            pyxel.rect(y * pixel, x * pixel, pixel, pixel, color)
    pyxel.rect(0, pixel, pixel, pixel, start_point_color)
    pyxel.rect((width - 1) * pixel, (height - 2) * pixel, pixel, pixel, end_point_color)

    if self.index > 0:
        # draw route
        offset = pixel / 2
        for i in range(len(self.route) - 1):
            curr = self.route[i]
            next = self.route[i + 1]
            self.color = backtrack_color if curr in self.route[:i] and next in self.route[:i] else route_color
            pyxel.line(curr[0] + offset, (curr[1] + offset), next[0] + offset, next[1] + offset, self.color)
        pyxel.circ(self.route[-1][0] + 2, self.route[-1][1] + 2, 1, head_color)

def update(self):
    if pyxel.btn(pyxel.KEY_Q):
        pyxel.quit()

    if pyxel.btn(pyxel.KEY_S):
        self.death = False

    if not self.death:
        self.check_death()
        self.update_route()

def check_death(self):
    if self.dfs_model and len(self.route) == len(self.dfs_route) - 1:
        self.death = True
    elif not self.dfs_model and len(self.route) == len(self.bfs_route) - 1:
        self.death = True

def update_route(self):
    index = int(self.index / self.step)
    self.index += 1
    if index == len(self.route):  # move
        if self.dfs_model:
            self.route.append([pixel * self.dfs_route[index][0], pixel * self.dfs_route[index][1]])
        else:
            self.route.append([pixel * self.bfs_route[index][0], pixel * self.bfs_route[index][1]])

App()

至此,我們完整的從迷宮生成,到尋找路徑,再到路徑可視化已全部實現。直接調用主函數 App() 然後按 S 鍵盤開啟遊戲,就可以看到文首的效果了。

總結

今天我們用深度優先算法實現了迷宮的遍歷,對於新手來說,遞歸這思路可能比較難理解,但這才是符合計算機思維的,隨著經驗的加深會理解越來越深刻的。

其次我們用 pyxel 庫來實現路徑可視化,難點在於坐標的計算更新,細節比較多且繁瑣,當然讀者也可以用其他庫或者直接用網頁來實現也可以。

遊戲源碼:

https://github.com/JustDoPython/python-examples/blob/master/doudou/2020-06-12-maze/maze.py

快來一試身手吧。

如果文章對你有幫助,歡迎轉發/點讚/收藏~

_往期文章推薦_

相關焦點

  • Python一行代碼,能玩這麼多童年的遊戲?
    安裝與使用安裝當然也很簡單一行代碼就可以pip install freegames由於該項目中的所有遊戲均是基於Python內置模塊Turtle製作,所以沒有太多依賴,安裝不會有困難。>貪吃蛇的玩法想必不用過多解釋了,使用鍵盤即可操控吃豆人吃豆人沒玩過也應該聽過,使用下面的代碼可以啟動一個類似吃豆人的遊戲python -m freegames.pacmanFlappy
  • turtle製作遊戲秘籍之一
    這可讓想用Python海龜畫圖模塊製作遊戲的小夥伴們情以何勘!Python海龜畫圖模塊並沒有提供讓海龜自毀的機制。你可以用Turtle這個類來新建一個叫tom的海龜,然後用del命令刪除tom這個名字,但是不要期待Python的內存自動管理機制會徹底刪除創建海龜的時候所留下的各種「遺蹟「。這些「遺蹟「是什麼呢?
  • 50行Python代碼實現經典遊戲,不僅是划水神器,更是學習利器!
    --Terri FurtonFree Python Games在輕鬆的環境中把遊戲和學習結合在一起,從而減輕了編程過程中的壓力。--Brett Bymaster...貪吃蛇、迷宮、吃豆人、掃雷、Flappy Bird...這些遊戲可以是非常經典,甚至伴隨著很多人的童年回憶。那麼,你是否想過自己開發一款專屬遊戲?
  • 製作一個猜數字的遊戲
    十一節假日,我在敲代碼,外甥女突然問我,「舅舅,你能不能給我編個遊戲啊」。
  • 原創 | 整理了38個Python遊戲開發庫
    拓展:對Pygame感興趣的建議看一下網站內的黑猩猩教程例子,網站直達:https://www.pygame.org/docs/tut/ChimpLineByLine.html所有主要的Panda3D應用程式都是用Python編寫的,這是使用該引擎的預期方式。Panda3D現在支持自動著色器生成,這意味著您可以使用法線貼圖、光澤度貼圖、光暈貼圖、HDR、卡通著色等,而無需編寫任何著色器。Panda3D還是一個現代引擎,支持高級功能,如著色器、模具和渲染到紋理。Panda3D與眾不同之處在於它強調短的學習曲線、快速的開發以及極端的穩定性和健壯性。P
  • 用Python製作一個貓咪小秒表
    秒表是一項隨處可見的神奇小物件,最常用到秒表的兩大場景,一個是運動會,另一個是健身房,因此也總是讓人聯想到汗水和心跳,賁張的血管,
  • 用CD盒給寶貝做一個迷宮
    千萬別,這些正方形的容器可是好東西,它是你製作小迷宮玩具的最好材料。只要有一點小孩子玩的粘土就可以搞定了!主要材料:空CD盒2個蠟線1包 (多色)軟陶土100克彩紙3張 A4所需工具:製作步驟:第1步:
  • 用 Python 寫一個安卓 APP
    http://youerning.blog.51cto.com/10513771/1733534前言用 Python 寫安卓 APP 肯定不是最好的選擇,目前用Java和 kotlin 寫的居多,但是肯定也是一個很偷懶的選擇
  • 《迷宮傳說》遊戲評測:像素風Rougelike遊戲
    《迷宮傳說》是一款像素風冒險遊戲,其中也帶著Rougelike自由玩法,遊戲簡化養成的需要,玩家不用把全部精力都放在收集養成這方面,而是更多的關注戰鬥操作和意識,這樣較為新穎的設定,也讓遊戲耳目一新。
  • 用Python寫一個安卓APP
    Python 寫安卓 APP 肯定不是最好的選擇,目前用 Java 和 kotlin 寫的居多,但是肯定也是一個很偷懶的選擇,而且實在不想學習 Java,再者,就編程而言已經會的就 Python與 Golang(註:Python,Golang 水平都一般),那麼久 Google了一下 Python 寫安卓的 APP 的可能性,還真行。
  • 【繪畫創意】迷宮遊戲大作戰,為什麼兒童特別熱衷於畫迷宮
    迷宮指的是充滿複雜通道,很難找到從其內部到達入口或從入口到達中心的道路,道路複雜難辨,人進去不容易出來的建築物。
  • Python程序突破微信小遊戲」跳一跳「
    我反應比較慢,上周看到朋友推薦了這個遊戲,但是我沒有及時更新微信,所以完不了。前天更新了一下微信,然後手機測試了一把,奈何技術太差,只打出了17分的成績。昨天晚上推薦給老婆玩,一下子被老婆秒殺了。老婆輕輕鬆鬆把分數推進到52分。於是我朋友圈求攻略,由此發現了python突破跳一跳小遊戲的方案。今天早上親自測試了一把,實戰通過。故此寫下攻略,與大家分享。
  • 迷宮很難玩?設計迷宮更難!!!力旺家創工作坊教你如何設計迷宮~
    用Scratch設計迷宮遊戲 迷宮遊戲是培養孩子空間能力的一種非常好的方式。當孩子進行迷宮遊戲時,研究者發現孩子大腦裡記錄空間關係的頂葉區變得活躍。 簡言之,如果孩子經常玩迷宮遊戲,能鍛鍊孩子大腦在空間定位方面的能力。那些經常玩迷宮遊戲的孩子,被發現在圖形或空間旋轉的測試題方面有更優秀的表現。孩子經常集中注意力做一件事情,他們的專注力會加速地提高。
  • 如何用 Python 寫一個 Discord 機器人
    在本教程中,您將學習如何使用 Python創建一個簡單的 Discord 機器人。也許您還不知道什麼是 Discord,本質上它是一項針對遊戲玩家的一種類 Slack(一個雲協作團隊工具和服務)的服務。在 Discord 上,您可以連接多個伺服器,您一定也注意到這些伺服器有許多機器人。
  • Python3.6實戰製作時鐘
    我們工作中經常需要觀看時間,我們今天就製作一個時鐘。代碼如下。界面的話,我們使用python自帶的tk模塊進行設計。關於strftime方法的使用,我們可以參考python幫助文件中的文檔說明。
  • 《磚塊迷宮建造者》怎麼建造迷宮 自建迷宮方法教學
    導 讀 磚塊迷宮建造者擁有好玩的自製迷宮功能,很多磚塊迷宮建造者新玩家不懂如何自製迷宮,怎麼開放迷宮給其他人玩
  • 用ExcelPython在Excel中調用Python
    這是一個非常簡單的python任務,只需要幾行代碼,而如果用VBA代碼來實現同樣功能則需要更多的代碼。也就是說現在VBA函數擁有了一個本地的叫methods的變量,它直接調用的其實是python模塊,注意AddPath參數,這裡要填入我們一開始定義python腳本的路徑-這裡我們輸入對應路徑以便找到我們的Methods.py。
  • 迷宮遊戲?磁力片積木還能這樣玩
    今天,我想跟大家分享的是,磁力片還可以玩迷宮遊戲哦。這個遊戲,有助於孩子分辨顏色,對指令不能更好理解的小朋友,特別是小一點的寶寶,讓他選擇單一顏色的路線,在難度和技術上都對寶寶或許是一個挑戰,但同時也是一個更為具體、明確、利於執行的指令。
  • 學python?不是一個python入門教程就行,學之前你必須知道這些
    機器學習:這也是python最有魅力的地方,善於做圖形分析,算法建模等等。所以python在人工智慧,機器學習的領域有著讀到的優勢。既然是就業那麼就要看市場,就是人才需求市場,這裡說的市場當讓是說python人才需求的市場了。說到市場當然python每個方向肯定有市場了,咱們直接看主要矛盾:一個是需求量,另一個是入行的難易程度。python全棧目前是市場的需求量最大,入行也是最容易的。要是為了就業那就先這樣入門入行,就不用想了。看重前景方向:那麼學python大數據分析或是python機器學習。
  • wind爸爸:5分鐘設計迷宮遊戲
    今天我們開始聊如何和孩子一起自己設計更好玩的迷宮遊戲。自己設計迷宮遊戲我把它分為紙上迷宮、DIY迷宮兩大類,下面與你分享我的實踐設計經驗,文末會附上素材,大家有需要可自行down。設計方法:第一步 畫外牆以正方形為例,先在紙上用鉛筆畫一個正方形,然後在距離相對遠的任意兩個位置,用橡皮擦擦出兩個空白,這兩個空白將作為起點和終點。通常我會選擇兩個角來擦。第二步 畫路線在外牆內開始畫更多的路線,建議每條路線上都擦出至少2個以上的空白,每條路線上的空白都上外面一層錯開。