欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

基于Python的A*算法解決八數(shù)碼問題實(shí)現(xiàn)步驟

 更新時(shí)間:2024年11月20日 10:54:22   作者:喜歡躺平劃水摸魚的擺爛貓  
這篇文章主要給大家介紹了關(guān)于如何基于Python的A*算法解決八數(shù)碼問題的實(shí)現(xiàn)步驟,文中介紹了八數(shù)碼問題及其求解方法,通過啟發(fā)式搜索算法,特別是A*算法,可以有效地解決八數(shù)碼問題,,需要的朋友可以參考下

一、問題描述

八數(shù)碼問題是人工智能領(lǐng)域一個(gè)經(jīng)典的問題。也是我們所熟悉的最簡(jiǎn)單的3×3數(shù)字華容道游戲:在一個(gè)3×3的九宮格棋盤上,擺有8個(gè)正方形方塊,每一個(gè)方塊都標(biāo)有1~8中的某一個(gè)數(shù)字。棋盤中留有一個(gè)空格,要求按照每次只能將與空格相鄰的方塊與空格交換的原則,將任意擺放的數(shù)碼盤(初始狀態(tài))逐步擺成某種給定的數(shù)碼盤的排列方式(目標(biāo)狀態(tài))。

二、涉及算法

啟發(fā)式搜索又稱為有信息搜索,是利用問題擁有啟發(fā)信息引導(dǎo)搜索,以達(dá)到減小搜索范圍、降低問題復(fù)雜度的目的。在啟發(fā)式搜索過程中,要對(duì)Open表進(jìn)行排序,這就要有一種方法來計(jì)算待擴(kuò)展結(jié)點(diǎn)有希望通向目標(biāo)結(jié)點(diǎn)的不同程度,人們總是希望找到最有可能通向目標(biāo)結(jié)點(diǎn)的待擴(kuò)展結(jié)點(diǎn)優(yōu)先擴(kuò)展。一種最常用的方法是定義一個(gè)評(píng)價(jià)函數(shù)對(duì)各個(gè)結(jié)點(diǎn)進(jìn)行計(jì)算,其目的就是用來估算出“有希望”的結(jié)點(diǎn)。用f來標(biāo)記評(píng)價(jià)函數(shù),用f(n)表示結(jié)點(diǎn)n的評(píng)價(jià)函數(shù)值,并用f來排列等待擴(kuò)展的結(jié)點(diǎn),然后選擇具有最小f值的結(jié)點(diǎn)作為下一個(gè)要擴(kuò)展的結(jié)點(diǎn)。

A*算法是一種有序搜索算法,其特點(diǎn)在于對(duì)評(píng)價(jià)函數(shù)的定義上。這個(gè)評(píng)估函數(shù)f使得在任意結(jié)點(diǎn)上其函數(shù)值f(n)能估算出結(jié)點(diǎn)S到結(jié)點(diǎn)n的最小代價(jià)路徑的代價(jià)與從節(jié)點(diǎn)n到某一目標(biāo)節(jié)點(diǎn)的最小代價(jià)路徑的代價(jià)的總和,也就是說f(n)是約束通過結(jié)點(diǎn)n的一條最小代價(jià)路徑的代價(jià)的估計(jì)。
算法具體內(nèi)容見文獻(xiàn):

https://wenku.baidu.com/view/4a80a40fa2161479171128de?_wkts_=1713600420821

三、實(shí)現(xiàn)步驟

在運(yùn)行前,需要提前準(zhǔn)備好infile.txt文件,第一行規(guī)定N的大小,即棋盤的大小,第二行則放置起始狀態(tài)棋盤中的數(shù)字排列,從上往下,從左往右一次排成一列,空格置為0。

1.定義狀態(tài)結(jié)點(diǎn)的類

定義一個(gè)State類,主要用于表示搜索過程中的狀態(tài)結(jié)點(diǎn),包括結(jié)點(diǎn)的代價(jià)和狀態(tài)信息,以及結(jié)點(diǎn)之間的關(guān)系。

屬性:

  • gn:從起始結(jié)點(diǎn)到當(dāng)前結(jié)點(diǎn)的實(shí)際代價(jià)。
  • hn:從當(dāng)前結(jié)點(diǎn)到目標(biāo)結(jié)點(diǎn)的估計(jì)代價(jià)(啟發(fā)式函數(shù))。
  • fn:綜合代價(jià),即gn+hn。
  • child:子結(jié)點(diǎn)列表,存儲(chǔ)從當(dāng)前結(jié)點(diǎn)可以到達(dá)的所有子結(jié)點(diǎn)。
  • par:父結(jié)點(diǎn),指向生成當(dāng)前結(jié)點(diǎn)的父結(jié)點(diǎn)。
  • state:當(dāng)前結(jié)點(diǎn)的狀態(tài)矩陣。
  • hash_value:當(dāng)前結(jié)點(diǎn)狀態(tài)矩陣的哈希值,用于在查找表中快速查找。

方法:

  • __lt__:小于運(yùn)算符重載,用于結(jié)點(diǎn)比較。
  • __eq__:等于運(yùn)算符重載,用于結(jié)點(diǎn)比較。
  • __ne__:不等于運(yùn)算符重載,用于結(jié)點(diǎn)比較。
class State(object):
    def __init__(self, gn=0, hn=0, state=None, hash_value=None, par=None):
        self.gn = gn
        self.hn = hn
        self.fn = self.gn + self.hn
        self.child = []
        self.par = par
        self.state = state
        self.hash_value = hash_value

    def __lt__(self, other):
        return self.fn < other.fn

    def __eq__(self, other):
        return self.hash_value == other.hash_value

    def __ne__(self, other):
        return not self.__eq__(other)

2.定義曼哈頓距離計(jì)算函數(shù)

計(jì)算兩個(gè)狀態(tài)結(jié)點(diǎn)之間的曼哈頓距離,作為啟發(fā)式函數(shù)的一部分,用于評(píng)估當(dāng)前結(jié)點(diǎn)到目標(biāo)結(jié)點(diǎn)的估計(jì)代價(jià)。

def manhattan_dis(cur_node, end_node):  # 定義一個(gè)名為manhattan_dis的函數(shù),接受兩個(gè)參數(shù)cur_node(當(dāng)前結(jié)點(diǎn))和end_node(目標(biāo)結(jié)點(diǎn))
    # 獲取當(dāng)前結(jié)點(diǎn)和目標(biāo)結(jié)點(diǎn)的狀態(tài)矩陣
    cur_state = cur_node.state
    end_state = end_node.state
    dist = 0
    N = len(cur_state)  # 獲取狀態(tài)矩陣的大小,假設(shè)為N
    # 遍歷狀態(tài)矩陣中的每個(gè)位置
    for i in range(N):
        for j in range(N):
            # 如果當(dāng)前結(jié)點(diǎn)的值與目標(biāo)結(jié)點(diǎn)的值相等,則跳過當(dāng)前位置,因?yàn)檫@個(gè)位置已經(jīng)在目標(biāo)狀態(tài)中
            if cur_state[i][j] == end_state[i][j]:
                continue
            num = cur_state[i][j]  # 獲取當(dāng)前結(jié)點(diǎn)在狀態(tài)矩陣中的值
            # 如果當(dāng)前結(jié)點(diǎn)的值為0(空白格),則將目標(biāo)位置設(shè)置為狀態(tài)矩陣的右下角
            if num == 0:
                x = N - 1
                y = N - 1
            # 如果當(dāng)前結(jié)點(diǎn)的值不為0,則根據(jù)當(dāng)前結(jié)點(diǎn)的值計(jì)算其目標(biāo)位置,假設(shè)目標(biāo)位置為(x,y)
            else:
                x = num / N
                y = num - N * x - 1
            # 計(jì)算當(dāng)前結(jié)點(diǎn)與目標(biāo)位置之間的曼哈頓距離,并累加到總距離中
            dist += (abs(x - i) + abs(y - j))
    # 返回計(jì)算得到的曼哈頓距離作為當(dāng)前結(jié)點(diǎn)到目標(biāo)結(jié)點(diǎn)的估計(jì)代價(jià)
    return dist

3.預(yù)留占位函數(shù)

test_fn是一個(gè)占位函數(shù),接受當(dāng)前結(jié)點(diǎn)和目標(biāo)結(jié)點(diǎn)作為參數(shù)。目前這個(gè)函數(shù)沒有實(shí)際的功能。

def test_fn(cur_node, end_node):
    return 0

4.生成子結(jié)點(diǎn)函數(shù)

創(chuàng)建generate_child函數(shù),接受當(dāng)前結(jié)點(diǎn)cur_node、目標(biāo)結(jié)點(diǎn)end_node、哈希集合hash_set、OPEN表open_table和距離函數(shù)dis_fn作為參數(shù)。實(shí)現(xiàn)了在當(dāng)前結(jié)點(diǎn)基礎(chǔ)上生成可行的子結(jié)點(diǎn),并考慮了重復(fù)狀態(tài)的處理,是A*算法中搜索過程的重要一步。

def generate_child(cur_node, end_node, hash_set, open_table, dis_fn):
    # 如果當(dāng)前結(jié)點(diǎn)就是目標(biāo)結(jié)點(diǎn),則直接將目標(biāo)結(jié)點(diǎn)假如OPEN表,并返回,表示已經(jīng)找到了解
    if cur_node == end_node:
        heapq.heappush(open_table, end_node)
        return
    # 獲取當(dāng)前結(jié)點(diǎn)狀態(tài)矩陣的大小
    num = len(cur_node.state)
    # 遍歷當(dāng)前結(jié)點(diǎn)狀態(tài)矩陣的每一個(gè)位置
    for i in range(0, num):
        for j in range(0, num):
            # 如果當(dāng)前位置不是空格,則跳過,因?yàn)榭崭袷强梢砸苿?dòng)的位置
            if cur_node.state[i][j] != 0:
                continue
            # 遍歷當(dāng)前位置的四個(gè)鄰居位置,即上下左右四個(gè)方向
            for d in direction:
                x = i + d[0]
                y = j + d[1]
                if x < 0 or x >= num or y < 0 or y >=num:
                    continue
                # 記錄生成的結(jié)點(diǎn)數(shù)量
                global SUM_NODE_NUM
                SUM_NODE_NUM += 1
                # 交換空格和鄰居位置的數(shù)字,生成一個(gè)新的狀態(tài)矩陣
                state = copy.deepcopy(cur_node.state)
                state[i][j], state[x][y] = state[x][y], state[i][j]
                # 計(jì)算新狀態(tài)矩陣的哈希值,并檢查是否已經(jīng)在哈希集合中存在,如果存在則表示已經(jīng)生成過相同的狀態(tài),跳過
                h = hash(str(state))
                if h in hash_set:
                    continue
                # 將新狀態(tài)的哈希值添加到哈希集合中,計(jì)算新狀態(tài)結(jié)點(diǎn)的gn(從起始結(jié)點(diǎn)到當(dāng)前結(jié)點(diǎn)的代價(jià))和hn(當(dāng)前結(jié)點(diǎn)到目標(biāo)結(jié)點(diǎn)的估計(jì)代價(jià))
                hash_set.add(h)
                gn = cur_node.gn + 1
                hn = dis_fn(cur_node, end_node)
                # 創(chuàng)建新的狀態(tài)結(jié)點(diǎn)對(duì)象,并將其加入到當(dāng)前結(jié)點(diǎn)的子結(jié)點(diǎn)列表中,并將其加入到OPEN表中。
                node = State(gn, hn, state, h, cur_node)
                cur_node.child.append(node)
                heapq.heappush(open_table, node)

5.定義輸出路徑函數(shù)

定義了一個(gè)名為print_path的函數(shù),接受一個(gè)參數(shù)node,表示目標(biāo)結(jié)點(diǎn)。通過回溯父結(jié)點(diǎn)的方式,從目標(biāo)結(jié)點(diǎn)一直回溯到起始結(jié)點(diǎn),并將沿途經(jīng)過的狀態(tài)矩陣打印出來,以展示搜索路徑。

def print_path(node):
    # 獲取從起始結(jié)點(diǎn)到目標(biāo)結(jié)點(diǎn)的路徑長(zhǎng)度,即目標(biāo)結(jié)點(diǎn)的實(shí)際代價(jià)
    num = node.gn
    
    # 定義了一個(gè)內(nèi)部函數(shù)show_block,用于打印狀態(tài)矩陣
    def show_block(block):
        print("---------------")
        for b in block:
            print(b)
    # 創(chuàng)建一個(gè)棧,用于存儲(chǔ)路徑中經(jīng)過的結(jié)點(diǎn)
    stack = []
    # 從目標(biāo)結(jié)點(diǎn)開始,沿著父結(jié)點(diǎn)指針一直回溯到起始結(jié)點(diǎn),并將沿途經(jīng)過的狀態(tài)矩陣入棧
    while node.par is not None:
        stack.append(node.state)
        node = node.par
    stack.append(node.state)
    # 從棧中依次取出狀態(tài)矩陣,并打印出來
    while len(stack) != 0:
        t = stack.pop()
        show_block(t)
    # 返回路徑長(zhǎng)度
    return num

6.定義A*算法

定義A_start函數(shù),接受起始狀態(tài)start、目標(biāo)狀態(tài)end、距離函數(shù)distance_fn、生成子結(jié)點(diǎn)函數(shù)generate_child_fn和可選的時(shí)間限制time_limit作為參數(shù)。實(shí)現(xiàn)了A*算法的整個(gè)搜索過程,包括結(jié)點(diǎn)的擴(kuò)展、路徑的搜索和時(shí)間限制的處理。

def A_start(start, end, distance_fn, generate_child_fn, time_limit=10):
    # 創(chuàng)建起始狀態(tài)結(jié)點(diǎn)和目標(biāo)狀態(tài)結(jié)點(diǎn)對(duì)象,并分別計(jì)算其哈希值
    root = State(0, 0, start, hash(str(BLOCK)), None)
    end_state = State(0, 0, end, hash(str(GOAL)), None)
    # 檢查起始狀態(tài)是否就是目標(biāo)狀態(tài),如果是,則直接輸出提示信息
    if root == end_state:
        print("start == end !")
    # 將起始狀態(tài)結(jié)點(diǎn)加入到OPEN表中,并對(duì)OPEN表進(jìn)行堆化操作
    OPEN.append(root)
    heapq.heapify(OPEN)
    # 創(chuàng)建一個(gè)哈希集合,用于存儲(chǔ)已經(jīng)生成的狀態(tài)結(jié)點(diǎn)的哈希值,并將起始狀態(tài)結(jié)點(diǎn)的哈希值添加到集合中
    node_hash_set = set()
    node_hash_set.add(root.hash_value)
    # 記錄算法開始的時(shí)間
    start_time = datetime.datetime.now()
    # 進(jìn)入主循環(huán),直到OPEN表為空(搜索完成)或達(dá)到時(shí)間限制
    while len(OPEN) != 0:
        top = heapq.heappop(OPEN)
        # 如果當(dāng)前結(jié)點(diǎn)就是目標(biāo)狀態(tài)結(jié)點(diǎn),則直接輸出路徑
        if top == end_state:
            return print_path(top)
        # 產(chǎn)生孩子節(jié)點(diǎn),孩子節(jié)點(diǎn)加入OPEN表
        generate_child_fn(cur_node=top, end_node=end_state, hash_set=node_hash_set,
                          open_table=OPEN, dis_fn=distance_fn)
        # 記錄當(dāng)前時(shí)間
        cur_time = datetime.datetime.now()
        # 超時(shí)處理,如果運(yùn)行時(shí)間超過了設(shè)定的時(shí)間限制,則輸出超時(shí)提示信息并返回
        if (cur_time - start_time).seconds > time_limit:
            print("Time running out, break !")
            print("Number of nodes:", SUM_NODE_NUM)
            return -1
    # 如果循環(huán)結(jié)束時(shí)OPEN表為空,則表示沒有找到路徑,輸出提示信息并返回-1
    print("No road !")  # 沒有路徑
    return -1

7.讀取數(shù)據(jù)作為原始狀態(tài)

定義read_block函數(shù),接受三個(gè)參數(shù)block(狀態(tài)矩陣列表)、line(輸入的一行數(shù)據(jù))、N(狀態(tài)矩陣的大?。⑽谋緮?shù)據(jù)解析為狀態(tài)矩陣的形式,并存儲(chǔ)在列表中,為后續(xù)的狀態(tài)表示和求解提供原始數(shù)據(jù)。

def read_block(block, line, N):
    # 使用正則表達(dá)式提取輸入行中的數(shù)字?jǐn)?shù)據(jù),并存儲(chǔ)在列表res中
    pattern = re.compile(r'\d+')  # 正則表達(dá)式提取數(shù)據(jù)
    res = re.findall(pattern, line)
    # 初始化計(jì)數(shù)變量t和臨時(shí)列表tmp
    t = 0
    tmp = []
    # 遍歷提取的數(shù)字?jǐn)?shù)據(jù),將其轉(zhuǎn)換為整數(shù),并添加到臨時(shí)列表tmp中
    for i in res:
        t += 1
        tmp.append(int(i))
        # 當(dāng)計(jì)數(shù)變量t達(dá)到狀態(tài)矩陣的大小N時(shí),表示當(dāng)前行數(shù)據(jù)處理完畢,將臨時(shí)表添加到狀態(tài)矩陣列表中,并清空臨時(shí)表
        if t == N:
            t = 0
            block.append(tmp)
            tmp = []

8.定義主函數(shù)查看結(jié)果

通過主函數(shù)if __name__ == 'main'讀取輸入數(shù)據(jù)、調(diào)用A*算法求解八數(shù)碼問題,并輸出求解結(jié)果的相關(guān)信息。

if __name__ == '__main__':
    # 嘗試打開infile.txt文件,如果文件打開失敗,則輸出錯(cuò)誤信息并退出程序
    try:
        file = open('infile.txt', "r")
    except IOError:
        print("can not open file infile.txt !")
        exit(1)
    # 打開名為infile.txt文件,并將文件對(duì)象賦值給變量f
    f = open("infile.txt")
    # 讀取文件的第一行,獲取棋盤的大小NUMBER
    NUMBER = int(f.readline()[-2])
    # 根據(jù)棋盤大小生成目標(biāo)狀態(tài),并將目標(biāo)狀態(tài)存儲(chǔ)在列表GOAL中
    n = 1
    for i in range(NUMBER):
        l = []
        for j in range(NUMBER):
            l.append(n)
            n += 1
        GOAL.append(l)
    GOAL[NUMBER - 1][NUMBER - 1] = 0
    # 逐行讀取文件中的數(shù)據(jù)
    for line in f:  # 讀取每一行數(shù)據(jù)
        # 在每次處理新的輸入數(shù)據(jù)之前,需要清空OPEN表和BLOCK表
        OPEN = []
        BLOCK = []
        # 調(diào)用讀取數(shù)據(jù)的函數(shù),將當(dāng)前行的數(shù)據(jù)解析并存儲(chǔ)為狀態(tài)矩陣
        read_block(BLOCK, line, NUMBER)
        # 初始化生成的結(jié)點(diǎn)數(shù)量為0
        SUM_NODE_NUM = 0
        # 記錄算法開始的時(shí)間
        start_t = datetime.datetime.now()
        # 這里添加5秒超時(shí)處理,可以根據(jù)實(shí)際情況選擇啟發(fā)函數(shù)
        # 將求解路徑長(zhǎng)度存儲(chǔ)在length中
        length = A_start(BLOCK, GOAL, manhattan_dis, generate_child, time_limit=10)
        # 記錄算法結(jié)束時(shí)間
        end_t = datetime.datetime.now()
        # 如果找到了路徑,則輸出路徑長(zhǎng)度、算法執(zhí)行時(shí)間和生成的結(jié)點(diǎn)數(shù)量
        if length != -1:
            print("length =", length)
            print("time =", (end_t - start_t).total_seconds(), "s")
            print("Nodes =", SUM_NODE_NUM)

四、運(yùn)行結(jié)果

A*算法在解決八數(shù)碼問題中表現(xiàn)出較高的準(zhǔn)確性。通過啟發(fā)式函數(shù)曼哈頓距離的計(jì)算,能夠較準(zhǔn)確的評(píng)估當(dāng)前結(jié)點(diǎn)到目標(biāo)節(jié)點(diǎn)的代價(jià),并在搜索過程中選擇代價(jià)最小的路徑。通過和實(shí)際路徑長(zhǎng)度的比較,可以驗(yàn)證算法的準(zhǔn)確性。

同時(shí),A*算法在搜索過程中充分利用了啟發(fā)式函數(shù)的估計(jì)值,能夠更優(yōu)先的擴(kuò)展可能更接近目標(biāo)的結(jié)點(diǎn),從而提高搜索效率,但是,在某些復(fù)雜的情況下,仍可能耗費(fèi)較長(zhǎng)時(shí)間或無法找到解,這取決于問題的復(fù)雜度和啟發(fā)式函數(shù)的選擇。

以下結(jié)果顯示的是從初始狀態(tài)轉(zhuǎn)變成目標(biāo)狀態(tài)的一個(gè)具體過程。

---------------
[7, 2, 6]
[8, 1, 4]
[3, 5, 0]
---------------
[7, 2, 6]
[8, 1, 0]
[3, 5, 4]
---------------
[7, 2, 0]
[8, 1, 6]
[3, 5, 4]
---------------
[7, 0, 2]
[8, 1, 6]
[3, 5, 4]
---------------
[7, 1, 2]
[8, 0, 6]
[3, 5, 4]
---------------
[7, 1, 2]
[8, 5, 6]
[3, 0, 4]
---------------
[7, 1, 2]
[8, 5, 6]
[0, 3, 4]
---------------
[7, 1, 2]
[0, 5, 6]
[8, 3, 4]
---------------
[0, 1, 2]
[7, 5, 6]
[8, 3, 4]
---------------
[1, 0, 2]
[7, 5, 6]
[8, 3, 4]
---------------
[1, 5, 2]
[7, 0, 6]
[8, 3, 4]
---------------
[1, 5, 2]
[7, 3, 6]
[8, 0, 4]
---------------
[1, 5, 2]
[7, 3, 6]
[8, 4, 0]
---------------
[1, 5, 2]
[7, 3, 0]
[8, 4, 6]
---------------
[1, 5, 2]
[7, 0, 3]
[8, 4, 6]
---------------
[1, 5, 2]
[7, 4, 3]
[8, 0, 6]
---------------
[1, 5, 2]
[7, 4, 3]
[0, 8, 6]
---------------
[1, 5, 2]
[0, 4, 3]
[7, 8, 6]
---------------
[1, 5, 2]
[4, 0, 3]
[7, 8, 6]
---------------
[1, 0, 2]
[4, 5, 3]
[7, 8, 6]
---------------
[1, 2, 0]
[4, 5, 3]
[7, 8, 6]
---------------
[1, 2, 3]
[4, 5, 0]
[7, 8, 6]
---------------
[1, 2, 3]
[4, 5, 6]
[7, 8, 0]
length = 22
time = 0.09839 s
Nodes = 8274

進(jìn)程已結(jié)束,退出代碼為 0

五、完整代碼

import heapq
import copy
import re
import datetime

BLOCK = []
GOAL = []
direction = [[0, 1], [0, -1], [1, 0], [-1, 0]]
OPEN = []
SUM_NODE_NUM = 0

class State(object):
    def __init__(self, gn=0, hn=0, state=None, hash_value=None, par=None):
        self.gn = gn
        self.hn = hn
        self.fn = self.gn + self.hn
        self.child = []
        self.par = par
        self.state = state
        self.hash_value = hash_value

    def __lt__(self, other):
        return self.fn < other.fn

    def __eq__(self, other):
        return self.hash_value == other.hash_value

    def __ne__(self, other):
        return not self.__eq__(other)


def manhattan_dis(cur_node, end_node):
    cur_state = cur_node.state
    end_state = end_node.state
    dist = 0
    N = len(cur_state)
    for i in range(N):
        for j in range(N):
            if cur_state[i][j] == end_state[i][j]:
                continue
            num = cur_state[i][j]
            if num == 0:
                x = N - 1
                y = N - 1
            else:
                x = num / N
                y = num - N * x - 1
            dist += (abs(x - i) + abs(y - j))
    return dist


def test_fn(cur_node, end_node):
    return 0


def generate_child(cur_node, end_node, hash_set, open_table, dis_fn):
    if cur_node == end_node:
        heapq.heappush(open_table, end_node)
        return
    num = len(cur_node.state)
    for i in range(0, num):
        for j in range(0, num):
            if cur_node.state[i][j] != 0:
                continue
            for d in direction:
                x = i + d[0]
                y = j + d[1]
                if x < 0 or x >= num or y < 0 or y >= num:
                    continue
                global SUM_NODE_NUM
                SUM_NODE_NUM += 1
                state = copy.deepcopy(cur_node.state)
                state[i][j], state[x][y] = state[x][y], state[i][j]
                h = hash(str(state))
                if h in hash_set:
                    continue
                hash_set.add(h)
                gn = cur_node.gn + 1
                hn = dis_fn(cur_node, end_node)
                node = State(gn, hn, state, h, cur_node)
                cur_node.child.append(node)
                heapq.heappush(open_table, node)


def print_path(node):
    num = node.gn

    def show_block(block):
        print("---------------")
        for b in block:
            print(b)

    stack = []
    while node.par is not None:
        stack.append(node.state)
        node = node.par
    stack.append(node.state)
    while len(stack) != 0:
        t = stack.pop()
        show_block(t)
    return num

def A_start(start, end, distance_fn, generate_child_fn, time_limit=10):
    root = State(0, 0, start, hash(str(BLOCK)), None)
    end_state = State(0, 0, end, hash(str(GOAL)), None)
    if root == end_state:
        print("start == end !")
    OPEN.append(root)
    heapq.heapify(OPEN)
    node_hash_set = set()
    node_hash_set.add(root.hash_value)
    start_time = datetime.datetime.now()
    while len(OPEN) != 0:
        top = heapq.heappop(OPEN)
        if top == end_state:
            return print_path(top)
        generate_child_fn(cur_node=top, end_node=end_state, hash_set=node_hash_set,
                          open_table=OPEN, dis_fn=distance_fn)
        cur_time = datetime.datetime.now()
        if (cur_time - start_time).seconds > time_limit:
            print("Time running out, break !")
            print("Number of nodes:", SUM_NODE_NUM)
            return -1
    print("No road !")
    return -1


def read_block(block, line, N):
    pattern = re.compile(r'\d+')
    res = re.findall(pattern, line)
    t = 0
    tmp = []
    for i in res:
        t += 1
        tmp.append(int(i))
        if t == N:
            t = 0
            block.append(tmp)
            tmp = []


if __name__ == '__main__':
    try:
        file = open('infile.txt', "r")
    except IOError:
        print("can not open file infile.txt !")
        exit(1)
    f = open("infile.txt")
    NUMBER = int(f.readline()[-2])
    n = 1
    for i in range(NUMBER):
        l = []
        for j in range(NUMBER):
            l.append(n)
            n += 1
        GOAL.append(l)
    GOAL[NUMBER - 1][NUMBER - 1] = 0
    for line in f:
        OPEN = []
        BLOCK = []
        read_block(BLOCK, line, NUMBER)
        SUM_NODE_NUM = 0
        start_t = datetime.datetime.now()
        length = A_start(BLOCK, GOAL, manhattan_dis, generate_child, time_limit=10)
        end_t = datetime.datetime.now()
        if length != -1:
            print("length =", length)
            print("time =", (end_t - start_t).total_seconds(), "s")
            print("Nodes =", SUM_NODE_NUM)

總結(jié) 

到此這篇關(guān)于基于Python的A*算法解決八數(shù)碼問題實(shí)現(xiàn)步驟的文章就介紹到這了,更多相關(guān) Python A*算法解決八數(shù)碼問題內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

最新評(píng)論