目的
data:image/s3,"s3://crabby-images/856e9/856e982de2a5f44542daba45f82ba3e128e256b3" alt="maze"
迷路の表現方法
各地点の表現
data:image/s3,"s3://crabby-images/695cb/695cb7fef337fc3e9bbdb4ae451315a19f2dd1d6" alt="maze-encode"
迷路自体の表現(モデル化)
data:image/s3,"s3://crabby-images/b23a8/b23a88edb8d1a3b3844eb5e89cc273a2b7c6f3fc" alt="maze-tree"
木とは
data:image/s3,"s3://crabby-images/0bbcf/0bbcfe858da33945bb2bbc6daeae999b633878aa" alt="tree-model"
深さ優先探索で迷路を解く
data:image/s3,"s3://crabby-images/1b173/1b1731632b127b4fd656b039db2c4154f361ff5a" alt="maze-depthfast1"
深さ優先探索とは以下の流れで各nodeを処理する方法である。
data:image/s3,"s3://crabby-images/0ae58/0ae58a07658e6a30dc04d7b5d1b0097c87ddcb3f" alt="maze-depthfast2"
data:image/s3,"s3://crabby-images/e40bf/e40bfd85d8e972374a75be3560f5a71ffc7a52f5" alt="maze-depthfast3"
data:image/s3,"s3://crabby-images/10efa/10efacf8c6f279e39b6ba99a0dc09244cc51420c" alt="maze-depthfast4"
data:image/s3,"s3://crabby-images/12e57/12e5787c4d9b9ef213023dcd4b2a2074338c469d" alt="maze-depthfast5"
data:image/s3,"s3://crabby-images/c73e3/c73e3867635341d65eea9266de0d8daad118164f" alt="maze-depthfast6"
data:image/s3,"s3://crabby-images/a519e/a519edce7a85494a890fe4d9324b35139d33a6cd" alt="maze-depthfast7"
data:image/s3,"s3://crabby-images/67e54/67e5424449ea4f17b5ac7b915d77a09d643b4c5f" alt="maze-depthfast8"
data:image/s3,"s3://crabby-images/0ae08/0ae0822bf14165743b98b1a27de9be3c7d22ed08" alt="maze-depthfast9"
data:image/s3,"s3://crabby-images/bd735/bd735aba914de6d7696e7a3daceda8cee4fab535" alt="maze-depthfast10"
data:image/s3,"s3://crabby-images/1f093/1f093ff7961f58133d9b4f1669ea443c9e914786" alt="maze-depthfast11"
data:image/s3,"s3://crabby-images/cb763/cb7637afaa2b35cad677a6b42744b8c2440ff2e4" alt="maze-depthfast12"
data:image/s3,"s3://crabby-images/92beb/92beb2ed0045dbb0d0adf7ce24be4589bb58d1cf" alt="maze-depthfast13"
プログラム
ルートの表示方法
data:image/s3,"s3://crabby-images/66919/669198fd3ae95de1698c468e8cbcb1a35f89937e" alt="maze-route"
コード
Go言語での実装package main import ( "bufio" "fmt" "os" ) type Point struct { X int Y int } type Content struct { Type string IsChecked bool } type Node struct { Parent *Node Pos Point } type Stack struct { Values []*Node Size int } func (s *Stack) Length() int { return s.Size } func (s *Stack) Push(n *Node) { s.Values[s.Size] = n s.Size += 1 } func (s *Stack) Pop() *Node { if s.Size > 0 { s.Size -= 1 return s.Values[s.Size] } return nil } func IsWay(maze [][]Content, x int, y int) bool { if x >= 0 && x <= 11 && y >= 0 && y <= 11 && maze[y][x].Type != "WALL" && maze[y][x].IsChecked == false { return true } else { return false } } var endNode *Node func SolveMaze(maze [][]Content, start Node) { stack := Stack{make([]*Node, 1000), 0} stack.Push(&start) cur := &start for ; maze[cur.Pos.Y][cur.Pos.X].Type != "END"; cur=stack.Pop() { fmt.Printf("(%v,%v) Stack Size : %v\n", cur.Pos.X, cur.Pos.Y, stack.Length()) maze[cur.Pos.Y][cur.Pos.X].IsChecked = true if IsWay(maze, cur.Pos.X, cur.Pos.Y-1) { child := Node{cur, Point{cur.Pos.X, cur.Pos.Y-1}} stack.Push(&child) } if IsWay(maze, cur.Pos.X, cur.Pos.Y+1) { child := Node{cur, Point{cur.Pos.X, cur.Pos.Y+1}} stack.Push(&child) } if IsWay(maze, cur.Pos.X-1, cur.Pos.Y) { child := Node{cur, Point{cur.Pos.X-1, cur.Pos.Y}} stack.Push(&child) } if IsWay(maze, cur.Pos.X+1, cur.Pos.Y) { child := Node{cur, Point{cur.Pos.X+1, cur.Pos.Y}} stack.Push(&child) } } endNode = cur } func main() { MAX_X := 11 MAX_Y := 11 // File open var fp *os.File fp, err := os.Open("maze.txt") if err != nil { panic(err) } defer fp.Close() // Convert maze string to array of point var startX, startY int maze := make([][]Content, MAX_Y+1) for y:= 0; y <= MAX_Y; y++ { maze[y] = make([]Content, MAX_X+1) } counter := 0 y := -1 scanner := bufio.NewScanner(fp) for scanner.Scan() { y += 1 x := -1 for _, char := range scanner.Text() { x += 1 switch char { case 'S': maze[y][x] = Content{"START", false} startX = x startY = y case 'G': maze[y][x] = Content{"END", false} case '.': maze[y][x] = Content{"WAY", false} case '#': maze[y][x] = Content{"WALL", false} } counter += 1 } } if err := scanner.Err(); err != nil { panic(err) } fmt.Println(maze) // Convert maze to binary tree start := Node{nil, Point{startX, startY}} SolveMaze(maze, start) fmt.Println(endNode) // Print Route cur := endNode for ; cur.Parent != nil; cur = cur.Parent { fmt.Println(cur.Pos) } fmt.Println(cur.Pos) }
実行結果
{9 11} # End {9 10} {8 10} {7 10} {6 10} {6 9} {6 8} {7 8} {8 8} {9 8} {9 7} {9 6} {8 6} {7 6} {6 6} {6 5} {6 4} {6 3} {6 2} {5 2} {4 2} {3 2} {2 2} {2 1} {2 0} # Start