<span id="mktg5"></span>

<i id="mktg5"><meter id="mktg5"></meter></i>

        <label id="mktg5"><meter id="mktg5"></meter></label>
        最新文章專題視頻專題問答1問答10問答100問答1000問答2000關鍵字專題1關鍵字專題50關鍵字專題500關鍵字專題1500TAG最新視頻文章推薦1 推薦3 推薦5 推薦7 推薦9 推薦11 推薦13 推薦15 推薦17 推薦19 推薦21 推薦23 推薦25 推薦27 推薦29 推薦31 推薦33 推薦35 推薦37視頻文章20視頻文章30視頻文章40視頻文章50視頻文章60 視頻文章70視頻文章80視頻文章90視頻文章100視頻文章120視頻文章140 視頻2關鍵字專題關鍵字專題tag2tag3文章專題文章專題2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章專題3
        問答文章1 問答文章501 問答文章1001 問答文章1501 問答文章2001 問答文章2501 問答文章3001 問答文章3501 問答文章4001 問答文章4501 問答文章5001 問答文章5501 問答文章6001 問答文章6501 問答文章7001 問答文章7501 問答文章8001 問答文章8501 問答文章9001 問答文章9501
        當前位置: 首頁 - 科技 - 知識百科 - 正文

        用Python代碼來解圖片迷宮的方法整理

        來源:懂視網 責編:小采 時間:2020-11-27 14:17:37
        文檔

        用Python代碼來解圖片迷宮的方法整理

        用Python代碼來解圖片迷宮的方法整理:譯注:原文是StackOverflow上一個如何用程序讀取迷宮圖片并求解的問題,幾位參與者熱烈地討論并給出了自己的代碼,涉及到用Python對圖片的處理以及廣度優先(BFS)算法等。 問題by Whymarrh: 當給定上面那樣一張JPEG圖片,如何才能更好地將這張圖轉換為合
        推薦度:
        導讀用Python代碼來解圖片迷宮的方法整理:譯注:原文是StackOverflow上一個如何用程序讀取迷宮圖片并求解的問題,幾位參與者熱烈地討論并給出了自己的代碼,涉及到用Python對圖片的處理以及廣度優先(BFS)算法等。 問題by Whymarrh: 當給定上面那樣一張JPEG圖片,如何才能更好地將這張圖轉換為合
        譯注:原文是StackOverflow上一個如何用程序讀取迷宮圖片并求解的問題,幾位參與者熱烈地討論并給出了自己的代碼,涉及到用Python對圖片的處理以及廣度優先(BFS)算法等。

        問題by Whymarrh:

        201542111941881.jpg (800×1002)

        當給定上面那樣一張JPEG圖片,如何才能更好地將這張圖轉換為合適的數據結構并且解出這個迷宮?

        我的第一直覺是將這張圖按像素逐個讀入,并存儲在一個包含布爾類型元素的列表或數組中,其中True代表白色像素,False代表非白色像素(或彩色可以被處理成二值圖像)。但是這種做法存在一個問題,那就是給定的圖片往往并不能完美的“像素化”。考慮到如果因為圖片轉換的原因,某個非預期的白色像素出現在迷宮的墻上,那么就可能會創造出一一條非預期的路徑。

        經過思考之后,我想出了另一種方法:首先將圖片轉換為一個可縮放適量圖形(SVG)文件,這個文件由一個畫布上的矢量線條列表組成,矢量線條按照列表的順序讀取,讀取出的仍是布爾值:其中True表示墻,而False表示可通過的區域。但是這種方法如果無法保證圖像能夠做到百分之百的精確轉換,尤其是如果不能將墻完全準確的連接,那么這個迷宮就可能出現裂縫。

        圖像轉換為SVG的另一個問題是,線條并不是完美的直線。因為SVG的線條是三次貝塞爾曲線,而使用整數索引的布爾值列表增加了曲線轉換的難度,迷宮線條上的所有點在曲線上都必須經過計算,但不一定能夠完美對應列表中的索引值。

        假設以上方法的確可以實現(雖然很可能都不行),但當給定一張很大的圖像時,它們還是不能勝任。那么是否存在一種更好地方法能夠平衡效率和復雜度?

        這就要討論到如何解迷宮了。如果我使用以上兩種方法中的任意一種,我最終將會得到一個矩陣。而根據這個問答(http://stackoverflow.com/questions/3097556/programming-theory-solve-a-maze/3097677#3097677),一個比較好的迷宮表示方式應該是使用樹的結構,并且使用A*搜索算法來解迷宮。那么如何從迷宮圖片中構造出迷宮樹呢?有比較好的方法么?

        以上廢話太多,總結起來問題就是:如何轉換迷宮圖片?轉換成為什么樣的數據結構?采用什么樣的數據結構能夠幫助或阻礙解迷宮?

        回答by Mikhail:

        這是我的解決方案:

        1. 將圖片轉換為灰度圖像(不是直接二值),調整不同顏色的權重使得最終的灰度看起來比較統一,你可以通過簡單地調節Photoshop 圖像->調整->黑白 菜單中的控制條來實現。
        2. 將上一步得到的灰度圖片轉換為二值圖片,可以通過在PS 圖像->調整->閾值 菜單中設定適當的閾值來實現
        3. 確保正確設置了閾值。使用魔棒工具(參數設置:容差 0、取樣點、連續以及消除鋸齒)選擇空白區域,檢查所選區域的邊緣不是因為錯誤的閾值設置而產生的假邊緣。事實上,這個迷宮中從start到end應該由聯通的空白區域。
        4. 人為地在迷宮外部加上邊界,確保迷宮漫游者^_^不會從start繞著迷宮跑到終點。:)
        5. 選擇語言實現廣度優先搜索算法(BFS),從start處開始讓程序運行。下面的代碼我選擇用Matlab實現。正如Thomas提到的,沒必要糾結于圖像的表示形式,你可以直接在二值圖像上運行。

        以下是用MATLAB實現的BFS代碼:

        function path = solve_maze(img_file)
         %% Init data
         img = imread(img_file);
         img = rgb2gray(img);
         maze = img > 0;
         start = [985 398];
         finish = [26 399];
         
         %% Init BFS
         n = numel(maze);
         Q = zeros(n, 2);
         M = zeros([size(maze) 2]);
         front = 0;
         back = 1;
         
         function push(p, d)
         q = p + d;
         if maze(q(1), q(2)) && M(q(1), q(2), 1) == 0
         front = front + 1;
         Q(front, :) = q;
         M(q(1), q(2), :) = reshape(p, [1 1 2]);
         end
         end
         
         push(start, [0 0]);
         
         d = [0 1; 0 -1; 1 0; -1 0];
         
         %% Run BFS
         while back <= front
         p = Q(back, :) ;
         back = back + 1;
         for i = 1:4
         push(p, d(i, :) );
         end
         end
         
         %% Extracting path
         path = finish;
         while true
         q = path(end, :) ;
         p = reshape(M(q(1), q(2), :) , 1, 2);
         path(end + 1, :) = p;
         if isequal(p, start)
         break;
         end
         end
        end
        
        

        這是個簡單的實現,應該很容易就能夠改寫為Python或其他語言,下面是程序的運行結果:

        201542112134749.jpg (800×1002)

        提問者更新:

        我用Python實現了一下Mikhail的方法,其中用到了numpy庫,感謝Thomas推薦。我感覺這個算法是正確的,但是效果不太如預期,以下是相關代碼,使用了PyPNG庫處理圖片。

        譯注:很遺憾,我用提問者提供的代碼并沒有跑通程序,并且似乎代碼縮進有點問題,而下面其他參與者的代碼能夠執行通過,并且效果很好。

        import png, numpy, Queue, operator, itertools
         
        def is_white(coord, image):
         """ Returns whether (x, y) is approx. a white pixel."""
         a = True
         for i in xrange(3):
         if not a: break
         a = image[coord[1]][coord[0] * 3 + i] > 240
         return a
         
        def bfs(s, e, i, visited):
         """ Perform a breadth-first search. """
         frontier = Queue.Queue()
         while s != e:
         for d in [(-1, 0), (0, -1), (1, 0), (0, 1)]:
         np = tuple(map(operator.add, s, d))
         if is_white(np, i) and np not in visited:
         frontier.put(np)
         visited.append(s)
         s = frontier.get()
         return visited
         
        def main():
         r = png.Reader(filename = "thescope-134.png")
         rows, cols, pixels, meta = r.asDirect()
         assert meta['planes'] == 3 # ensure the file is RGB
         image2d = numpy.vstack(itertools.imap(numpy.uint8, pixels))
         start, end = (402, 985), (398, 27)
         print bfs(start, end, image2d, [])
        
        

        回答by Joseph Kern:

        #!/usr/bin/env python
         
        import sys
         
        from Queue import Queue
        from PIL import Image
         
        start = (400,984)
        end = (398,25)
         
        def iswhite(value):
         if value == (255,255,255):
         return True
         
        def getadjacent(n):
         x,y = n
         return [(x-1,y),(x,y-1),(x+1,y),(x,y+1)]
         
        def BFS(start, end, pixels):
         
         queue = Queue()
         queue.put([start]) # Wrapping the start tuple in a list
         
         while not queue.empty():
         
         path = queue.get()
         pixel = path[-1]
         
         if pixel == end:
         return path
         
         for adjacent in getadjacent(pixel):
         x,y = adjacent
         if iswhite(pixels[x,y]):
         pixels[x,y] = (127,127,127) # see note
         new_path = list(path)
         new_path.append(adjacent)
         queue.put(new_path)
         
         print "Queue has been exhausted. No answer was found."
         
        if __name__ == '__main__':
         
         # invoke: python mazesolver.py [.jpg|.png|etc.]
         base_img = Image.open(sys.argv[1])
         base_pixels = base_img.load()
         
         path = BFS(start, end, base_pixels)
         
         path_img = Image.open(sys.argv[1])
         path_pixels = path_img.load()
         
         for position in path:
         x,y = position
         path_pixels[x,y] = (255,0,0) # red
         
         path_img.save(sys.argv[2])
        
        

        動態執行效果:

        201542112219406.gif (160×200)

        回答by Jim

        使用樹搜索太繁雜了,迷宮本身就跟解路徑是可分的。正因如此,你可以使用連通區域查找算法來標記迷宮中的連通區域,這將迭代搜索兩次這些像素點。如果你想要更好地解決方法,你可以對結構單元使用二元運算(binary operations)來填充每個連通區域中的死路。

        下面是相關的MATLAB代碼及運行結果:

        % read in and invert the image
        im = 255 - imread('maze.jpg');
         
        % sharpen it to address small fuzzy channels
        % threshold to binary 15%
        % run connected components
        result = bwlabel(im2bw(imfilter(im,fspecial('unsharp')),0.15));
         
        % purge small components (e.g. letters)
        for i = 1:max(reshape(result,1,1002*800))
         [count,~] = size(find(result==i));
         if count < 500
         result(result==i) = 0;
         end
        end
         
        % close dead-end channels
        closed = zeros(1002,800);
        for i = 1:max(reshape(result,1,1002*800))
         k = zeros(1002,800);
         k(result==i) = 1; k = imclose(k,strel('square',8));
         closed(k==1) = i;
        end
         
        % do output
        out = 255 - im;
        for x = 1:1002
         for y = 1:800
         if closed(x,y) == 0
         out(x,y,:) = 0;
         end
         end
        end
        imshow(out);
        
        

        201542112302746.jpg (964×1088)

        回答by Stefano

        stefano童鞋給出了生成搜索過程GIF及AVI文件的代碼 maze-solver-python (GitHub)

        201542112345448.gif (320×401)

        聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

        文檔

        用Python代碼來解圖片迷宮的方法整理

        用Python代碼來解圖片迷宮的方法整理:譯注:原文是StackOverflow上一個如何用程序讀取迷宮圖片并求解的問題,幾位參與者熱烈地討論并給出了自己的代碼,涉及到用Python對圖片的處理以及廣度優先(BFS)算法等。 問題by Whymarrh: 當給定上面那樣一張JPEG圖片,如何才能更好地將這張圖轉換為合
        推薦度:
        標簽: 代碼 程序 圖片的
        • 熱門焦點

        最新推薦

        猜你喜歡

        熱門推薦

        專題
        Top
        主站蜘蛛池模板: 日韩免费毛片视频| 噼里啪啦电影在线观看免费高清| 国产在线观看免费不卡| 亚洲人成网男女大片在线播放| 1a级毛片免费观看| 亚洲av无码专区在线| 久久久久久国产a免费观看黄色大片| 亚洲精品成人av在线| 最近2019中文字幕免费大全5| 亚洲国产精品人久久| 91av在线免费视频| 亚洲一卡2卡4卡5卡6卡残暴在线| 久草在视频免费福利| 亚洲日韩国产二区无码| 日本xxwwxxww在线视频免费| 亚洲AV无码精品国产成人| 免费人成在线观看视频播放| 一级女人18片毛片免费视频 | 在线成人a毛片免费播放| 亚洲欧美国产精品专区久久| 免费国产在线观看| 久久久精品视频免费观看| 亚洲成人激情在线| 免费看国产精品3a黄的视频| 亚洲av无码偷拍在线观看| 亚洲中文久久精品无码| 久久成人免费播放网站| 久久亚洲精品专区蓝色区| 免费在线黄色网址| 无码午夜成人1000部免费视频| 国产成人精品日本亚洲11| 亚洲伦乱亚洲h视频| 91短视频免费在线观看| 亚洲AV无码一区二区一二区| 亚洲中文字幕久久精品无码APP| 最近免费中文字幕大全免费版视频| 亚洲欧美日韩中文高清www777| 亚洲日本中文字幕一区二区三区| 久久w5ww成w人免费| 亚洲精华液一二三产区| 亚洲国产成人精品无码区在线观看|