摘要:實現(xiàn)這一功能最簡單的方法是計算棋盤上棋子的相對強度大小,用下面的對照表。本鏈接是基于算法優(yōu)化的國際象棋。我們來稍微調(diào)整一下棋盤上棋子狀態(tài)的權(quán)重,這一圖表是在國際象棋程序維基百科中給出的。
本文作者: Lauri Hartikka
編譯:胡子大哈翻譯原文:http://huziketang.com/blog/posts/
英文連接:A step-by-step guide to building a simple chess AI
轉(zhuǎn)載請注明出處,保留原文鏈接以及作者信息
首先讓我們先看幾個對開發(fā)簡單國際象棋 AI 很有幫助的概念:
移動生成
局面評估
極大極小算法
α-β 剪枝
每一步中我們都會對經(jīng)過時間檢驗的國際象棋程序進(jìn)行改進(jìn),我會展示不同算法風(fēng)格所產(chǎn)生的影響。你也可以在 GitHub 上看到最終的 AI 算法。
步驟 1:移動生成和棋盤可視化使用 chess.js 庫來生成移動規(guī)則,使用 chessboard.js 來可視化棋盤。移動生成庫實現(xiàn)了所有國際象棋的規(guī)則,對于任意給定的棋盤狀態(tài)我們都可以計算出下一步的合法的走棋方法。
(移動生成函數(shù)的可視化版本。起始位置作為輸入,輸出是所有可能的走法。)
使用這些庫可以使我們專注于我們所感興趣的任務(wù):開發(fā)最佳下棋的算法。我們首先從創(chuàng)建以一個函數(shù)開始,在所有可能走法中返回一個隨機(jī)的結(jié)果。
var calculateBestMove =function(game) { //generate all the moves for a given position var newGameMoves = game.ugly_moves(); return newGameMoves[Math.floor(Math.random() * newGameMoves.length)]; };
用這種方法,盡管它不是一個合格的棋手,但是起碼我們可以和它玩起來了。
步驟 2:位置評估下面我們試著讓它理解在一個確定的位置上怎么走比較好。實現(xiàn)這一功能最簡單的方法是計算棋盤上棋子的相對強度大小,用下面的對照表。
通過評估函數(shù),可以選擇評估結(jié)果最佳的走法。
var calculateBestMove = function (game) { var newGameMoves = game.ugly_moves(); var bestMove = null; //use any negative large number var bestValue = -9999; for (var i = 0; i < newGameMoves.length; i++) { var newGameMove = newGameMoves[i]; game.ugly_move(newGameMove); //take the negative as AI plays as black var boardValue = -evaluateBoard(game.board()) game.undo(); if (boardValue > bestValue) { bestValue = boardValue; bestMove = newGameMove } } return bestMove; };
這樣一來,一個切實的改善是,算法會吃掉它可以吃掉的棋子。
(黑子使用了簡單評估算法,在這里可以看到:https://jsfiddle.net/lhartikk...)
步驟 3:用極大極小算法搜索樹接下來我們來創(chuàng)建一個搜索樹,通過它算法可以選擇最佳走法,這里需要用到極大極小算法。
在這個算法中,會根據(jù)給定的樹深度對遞歸樹進(jìn)行遍歷,所要評估的狀態(tài)就是樹的葉子節(jié)點。
這一步完成以后我們把子節(jié)點中的最大或者最小值返回給父節(jié)點,這要依賴于白棋還是黑棋來走這一步(這就是說在樹的每一層中都最大或者最小化輸出)。
(給定狀態(tài)的最大最小算法的可視化。白棋最好的走法是 b2-c3,因為可以保證獲取一個狀態(tài)評估值是 -50)
var minimax = function (depth, game, isMaximisingPlayer) { if (depth === 0) { return -evaluateBoard(game.board()); } var newGameMoves = game.ugly_moves(); if (isMaximisingPlayer) { var bestMove = -9999; for (var i = 0; i < newGameMoves.length; i++) { game.ugly_move(newGameMoves[i]); bestMove = Math.max(bestMove, minimax(depth - 1, game, !isMaximisingPlayer)); game.undo(); } return bestMove; } else { var bestMove = 9999; for (var i = 0; i < newGameMoves.length; i++) { game.ugly_move(newGameMoves[i]); bestMove = Math.min(bestMove, minimax(depth - 1, game, !isMaximisingPlayer)); game.undo(); } return bestMove; } };
有了最大最小步驟以后,我們的算法可以下出一些國際象棋的基本策略了。
極大極小算法的效率取決于搜索樹的深度,這就是我們后面步驟要優(yōu)化的地方。
步驟 4:α-β 剪枝Alpha-beta 剪枝是極大極小算法的一種優(yōu)化方法,可以砍掉搜索樹中的某些分支。這可以幫助我們用同樣的資源的情況下,盡可能深地遍歷極大極小搜索樹。
α-β 剪枝的原理是在遍歷搜索樹的過程中發(fā)現(xiàn)可以終止遍歷的狀態(tài),進(jìn)而把整個分支剪掉的過程。這是因為發(fā)現(xiàn)下一步會導(dǎo)致比上一步更糟的結(jié)果,那么就不用再遍歷下去了。
α-β 剪枝不影響極大極小算法的結(jié)果,僅僅是使極大極小算法運行的更快。假設(shè)遍歷時恰巧第一個狀態(tài)就是最佳走法,那么 α-β 剪枝會更加有效。
有了 α-β,極大極小算法如虎添翼,可以看下面的例子。
(本圖是給定的起始棋盤狀態(tài),下面的數(shù)字是如果遍歷深度是 4 的話,需要評估的狀態(tài)總數(shù)。)
本鏈接是基于 α-β 算法優(yōu)化的國際象棋 AI。
步驟 5:改善評估函數(shù)初始的評估函數(shù)非常簡單,只是數(shù)了盤面上的數(shù)值而已。下面我們來改善它,把棋子的位置因素也考慮到評估結(jié)果里面去。例如在棋盤中間的馬會比在棋盤邊緣的馬位置更好(因為它的可選擇性更多,也更加活躍)。
我們來稍微調(diào)整一下棋盤上棋子狀態(tài)的權(quán)重,這一圖表是在國際象棋程序維基百科中給出的。
(棋盤權(quán)值表的可視化??梢愿鶕?jù)棋子的位置增加或者減少相應(yīng)位置的權(quán)重)
通過上面的一系列改進(jìn),我們的算法可以下出像樣的棋局了,起碼開始像一個業(yè)余棋手這樣了。
(改進(jìn)后的評估函數(shù)加上搜索樹深度設(shè)置成 3 的 α-β 算法,可以在這個地址看到:https://jsfiddle.net/q76uzxwe/1/)
總結(jié)這個簡單的國際象棋算法不會犯一些很傻的錯誤,但是它依然是缺乏策略理解的。
通過我所介紹的這種方法,可以開發(fā)一個國際象棋程序來實現(xiàn)一些基本的玩法?!癆I 部分”(不包括移動生成)只有 200 行代碼,也就是說這里只實現(xiàn)了基本的概念。你可以在 GitHub 中獲取最終版本的代碼。
關(guān)于算法的一些更深層次的改善可以見下面鏈接:
移動順序
快速移動生成策略
游戲結(jié)束策略評估
如果你想要了解更多,查看國際象棋程序維基百科,這里介紹了很多有用的資源,本文只是演示了國際象棋 AI 算法實現(xiàn)的基本步驟。
Happy Coding!如果本文對你有幫助,歡迎關(guān)注我的專欄-前端大哈,定期發(fā)布高質(zhì)量前端文章。
我最近正在寫一本《React.js 小書》,對 React.js 感興趣的童鞋,歡迎指點。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/82604.html
摘要:實現(xiàn)這一功能最簡單的方法是計算棋盤上棋子的相對強度大小,用下面的對照表。本鏈接是基于算法優(yōu)化的國際象棋。我們來稍微調(diào)整一下棋盤上棋子狀態(tài)的權(quán)重,這一圖表是在國際象棋程序維基百科中給出的。 showImg(https://segmentfault.com/img/remote/1460000009143081?w=2000&h=1317); 本文作者: Lauri Hartikka 編...
摘要:探探機(jī)器人,自動根據(jù)不同妹紙漢子顏值年齡等類型,喜歡忽略,歡迎各位先看一下實現(xiàn)的結(jié)果吧今天要講的主題是使用腳本實現(xiàn)你自己想要自動操控的任意手機(jī)。 前言 之前寫了篇文章:【全是干貨】談?wù)勅绾螌W(xué)習(xí)一項新技能,沒有理論,全是實戰(zhàn),里面第五點提到用腳本玩探探,昨天花了一個小時實現(xiàn)了該功能。 Github:探探機(jī)器人,自動根據(jù)不同妹紙/漢子顏值、年齡等類型,喜歡、忽略,歡迎各位star 先看一下...
摘要:實際前端開發(fā)中我們常常需要模擬數(shù)據(jù)市場上有許多工具提供使用但是基本都是提供數(shù)據(jù)展示如果我們想要一個具備增刪改查功能的接口怎么辦呢當(dāng)然是強大自己啦首先我們需要新建一個目錄安裝依賴及以上版本,首先確認(rèn)版本在以上,版本低的請自行搞定寫個試 實際前端開發(fā)中我們常常需要模擬數(shù)據(jù),市場上有許多工具提供使用但是基本都是提供數(shù)據(jù)展示,如果我們想要一個具備增刪改查功能的接口怎么辦呢?當(dāng)然是強大自己啦??!...
摘要:實際前端開發(fā)中我們常常需要模擬數(shù)據(jù)市場上有許多工具提供使用但是基本都是提供數(shù)據(jù)展示如果我們想要一個具備增刪改查功能的接口怎么辦呢當(dāng)然是強大自己啦首先我們需要新建一個目錄安裝依賴及以上版本,首先確認(rèn)版本在以上,版本低的請自行搞定寫個試 實際前端開發(fā)中我們常常需要模擬數(shù)據(jù),市場上有許多工具提供使用但是基本都是提供數(shù)據(jù)展示,如果我們想要一個具備增刪改查功能的接口怎么辦呢?當(dāng)然是強大自己啦!!...
閱讀 1238·2021-11-11 16:54
閱讀 887·2021-10-19 11:44
閱讀 1353·2021-09-22 15:18
閱讀 2456·2019-08-29 16:26
閱讀 2961·2019-08-29 13:57
閱讀 3106·2019-08-26 13:32
閱讀 1091·2019-08-26 11:58
閱讀 2340·2019-08-26 10:37