人工智能五子棋游戏——(7)完整的项目代码

本项目使用的是JavaScript语言,用到了其中的jquery库的jquery-2.2.2.min版本,请自行网上下载,本文就不再给出。

 

(1)前端html文件

index.html

 1 <!DOCTYPE html>
 2 <html>
 3 <head>
 4     <meta charset="utf-8" />
 5     <meta name="viewport" content="initial-scale=1.0,user-scalable=no,maximum-scale=1,width=device-width">
 6     <title>人工智能五子棋</title>
 7     <style type="text/css">
 8     body{
 9         margin: 0;
10         font-family: Microsoft Yahei;
11         text-align: center;
12     }
13     .timer{
14         margin: 0 20px;
15     }
16     .rollback{
17         width: 70px;
18         height: 30px;
19         margin: 20px;
20     }
21     .chessboard{
22         width: 315px;
23         margin: 0 auto;
24         border-left: 1px solid #ccc;
25         border-top: 1px solid #ccc;
26     }
27     .chessboard .row{
28         overflow: hidden;
29     }
30     .chessboard .tile{
31         width: 20px;
32         height: 20px;
33         float: left;
34         border-right: 1px solid #ccc;
35         border-bottom: 1px solid #ccc;
36         text-align: center;
37         line-height: 20px;
38     }
39     .chessboard .tile.current{
40         background-color: #ddd;
41     }
42     .chessboard .chess-max,.chessboard .chess-min{
43         font-size: 20px;
44         cursor: pointer;
45         -webkit-user-select: none;
46     }
47     </style>
48 </head>
49 <body>
50     <div>
51         <span class="timer">游戏开始,请落子</span>
52         <button class="rollback">悔棋</button>
53     </div>
54     <div class="chessboard"></div>
55     <script type="text/javascript" src="jquery-2.2.2.min.js"></script>
56     <script type="text/javascript" src="fivechess.js"></script>
57     <script type="text/javascript" src="main.js"></script>
58 </body>
59 </html>

 

(2)后端五子棋游戏的实现

main.js

 1 //棋盘
 2 var chessboard = new Chessboard(15, 15);
 3 var chessboardDom = $('.chessboard');
 4 //耗时和时钟
 5 var t = 0,
 6     tick;
 7 //初始化棋盘dom
 8 var initData = function() {
 9     for (var i = 0; i < chessboard.row; i++) {
10         var rowDom = $('<div></div>').addClass('row');
11         for (var j = 0; j < chessboard.column; j++) {
12             var tile = $('<div></div>').addClass('tile').data({
13                 row: i,
14                 column: j
15             });
16             rowDom.append(tile);
17         }
18         chessboardDom.append(rowDom);
19     }
20 };
21 var initEvent = function() {
22     $('.tile').on('click', function() {
23         //如果下棋还未完结
24         if (!chessboard.isEnded()) {
25             var self = $(this);
26             //已经有棋子
27             if (self.is('.chess')) {
28                 alert('请选择空余的格子下棋');
29             } else {
30                 var row = self.data('row'),
31                     column = self.data('column');
32                 //人下棋
33                 chessboard.put(row, column, Chessboard.MAX);
34                 self.addClass('chess chess-max current').text('●');
35                 //第一次下棋开始计时
36                 if (!tick) {
37                     tick = setInterval(function() {
38                         t++;
39                         $('.timer').text('对局已经过' + t + '秒');
40                     }, 1000);
41                 }
42                 //人赢了,结束游戏
43                 if (chessboard.isMaxWin()) {
44                     chessboard.end();
45                     clearInterval(tick);
46                     return alert('你用了' + t + '秒赢了人工智能');
47                 }
48                 //没赢,但平手了
49                 if (chessboard.isEnded()) {
50                     clearInterval(tick);
51                     return alert('你用了' + t + '秒和人工智能成为平手');
52                 } else {
53                     setTimeout(function() {
54                         //未分胜负,AI下棋
55                         console.time('min');
56                         var res = min(chessboard, 2);
57                         console.timeEnd('min');
58                         //AI下棋
59                         chessboard.put(res.row, res.column, Chessboard.MIN);
60                         $('.current').removeClass('current');
61                         chessboardDom.find('.row').eq(res.row).find('.tile').eq(res.column).addClass('chess chess-min current').text('○');
62                         //AI赢了,结束游戏
63                         if (chessboard.isMinWin()) {
64                             chessboard.end();
65                             clearInterval(tick);
66                             return alert('你被人工智能打败!');
67                         }
68                         //没赢,但平手了,否则未分胜负,等待人继续下棋
69                         if (chessboard.isEnded()) {
70                             clearInterval(tick);
71                             return alert('你用了' + t + '秒赢了人工智能');
72                         }
73                     }, 200);
74                 }
75             }
76         }
77     });
78     //悔棋
79     $('.rollback').on('click', function() {
80         //倒退两步
81         var steps = chessboard.rollback(2);
82         //清空前两步的dom文本
83         steps.forEach(function(step) {
84             chessboardDom.find('.row').eq(step.row).find('.tile').eq(step.column).removeClass('chess chess-min chess-max current').text('');
85         });
86         //返回当前的下棋位置,并添加current类
87         var step = chessboard.current();
88         if (step) {
89             chessboardDom.find('.row').eq(step.row).find('.tile').eq(step.column).addClass('current');
90         }
91     });
92 };
93 initData();
94 initEvent();

 

(2)算法的实现代码

fivechess.js

  1 /**
  2  * [Chessboard 棋盘]
  3  * @param {[type]} row    [description]
  4  * @param {[type]} column [description]
  5  */
  6 var Chessboard = function(row, column) {
  7     this.data = [];
  8     this.row = row;
  9     this.column = column;
 10     //赢法数组
 11     this.wins = [];
 12     //赢法数
 13     this.count = 0;
 14     //记录max每一种赢法的已经达成棋子数
 15     this.maxWin = [];
 16     //记录min每一种赢法的已经达成棋子数
 17     this.minWin = [];
 18     //初始化棋盘,顺便初始化赢法数组
 19     for (var i = 0; i < row; i++) {
 20         this.data[i] = [];
 21         this.wins[i] = [];
 22         for (var j = 0; j < column; j++) {
 23             this.data[i][j] = Chessboard.NONE;
 24             this.wins[i][j] = [];
 25         }
 26     }
 27     //横向赢法
 28     for (var i = 0; i < row; i++) {
 29         for (var j = 0; j <= column - 5; j++) {
 30             for (var k = 0; k < 5; k++) {
 31                 this.wins[i][j + k][this.count] = true;
 32             }
 33             this.count++;
 34         }
 35     }
 36     //纵向赢法
 37     for (var i = 0; i < column; i++) {
 38         for (var j = 0; j <= row - 5; j++) {
 39             for (var k = 0; k < 5; k++) {
 40                 this.wins[j + k][i][this.count] = true;
 41             }
 42             this.count++;
 43         }
 44     }
 45     //右斜线的赢法
 46     for (var i = 0; i <= row - 5; i++) {
 47         for (var j = 0; j <= column - 5; j++) {
 48             for (var k = 0; k < 5; k++) {
 49                 this.wins[i + k][j + k][this.count] = true;
 50             }
 51             this.count++;
 52         }
 53     }
 54     //左斜线的赢法
 55     for (var i = 0; i <= row - 5; i++) {
 56         for (var j = column - 1; j >= 4; j--) {
 57             for (var k = 0; k < 5; k++) {
 58                 this.wins[i + k][j - k][this.count] = true;
 59             }
 60             this.count++;
 61         }
 62     }
 63     //初始化max和min每一种赢法的下子情况
 64     for (var i = 0; i < this.count; i++) {
 65         this.maxWin[i] = {
 66             max: 0,
 67             min: 0
 68         };
 69         this.minWin[i] = {
 70             min: 0,
 71             max: 0
 72         };
 73     }
 74     //下棋记录堆栈
 75     this.stack = [];
 76     //游戏是否结束
 77     this.is_ended = false;
 78 };
 79 /**
 80  * [toString 输出棋盘信息]
 81  * @return {[type]} [description]
 82  */
 83 Chessboard.prototype.toString = function() {
 84     return this.data.map(function(data) {
 85         return data.toString();
 86     }).join('\n');
 87 };
 88 /**
 89  * 返回当前最新一步下的棋子
 90  * @return {[type]}
 91  */
 92 Chessboard.prototype.current = function() {
 93     var l = this.stack.length;
 94     if (l) {
 95         return this.stack[l - 1];
 96     }
 97 };
 98 /**
 99  * [put 下棋]
100  * @param  {[type]} row    [行]
101  * @param  {[type]} column [列]
102  * @param  {[type]} type   [人还是AI下棋]
103  * @return {[type]}        [description]
104  */
105 Chessboard.prototype.put = function(row, column, type) {
106     if (this.data[row][column] == Chessboard.NONE) {
107         this.data[row][column] = type;
108         //放进记录堆栈
109         this.stack.push({
110             row: row,
111             column: column,
112             type: type
113         });
114         //下棋之后对每一种赢法的下棋情况进行更新
115         for (var i = 0; i < this.count; i++) {
116             if (this.wins[row][column][i]) {
117                 if (type == Chessboard.MAX) {
118                     this.maxWin[i].max++;
119                     this.minWin[i].max++;
120                 } else {
121                     this.minWin[i].min++;
122                     this.maxWin[i].min++;
123                 }
124             }
125         }
126         //如果下子满了则结束游戏
127         if (this.stack.length == this.row * this.column) {
128             this.end();
129         }
130     }
131     return this;
132 };
133 /**
134  * [rollback 悔棋]
135  * @param  {[type]} n [后退n步]
136  * @return {[type]}   [description]
137  */
138 Chessboard.prototype.rollback = function(n) {
139     //记录后退的n步走法
140     var steps = [];
141     n = n || 1;
142     for (var i = 0; i < n; i++) {
143         var step = this.stack.pop();
144         if (step) {
145             steps.push(step);
146             var row = step.row,
147                 column = step.column,
148                 type = step.type;
149             //置空格点
150             this.data[row][column] = Chessboard.NONE;
151             //更新每一种赢法的下子情况
152             for (var j = 0; j < this.count; j++) {
153                 if (this.wins[row][column][j]) {
154                     if (type == Chessboard.MAX) {
155                         this.maxWin[j].max--;
156                         this.minWin[j].max--;
157                     } else {
158                         this.minWin[j].min--;
159                         this.maxWin[j].min--;
160                     }
161                 }
162             }
163         }
164     }
165     this.is_ended = false;
166     return steps;
167 };
168 /**
169  * 获取当前点附近考虑的下棋位置
170  * @param  {[p这个点]}
171  * @return {[type]}
172  */
173 Chessboard.prototype.getNearPoints = function(p) {
174     var points = [],
175         row, column;
176     for (var i = -2; i <= 2; i++) {
177         //右下角
178         row = p.row + i;
179         column = p.column + i;
180         if (this.isValid(row, column)) {
181             points.push({
182                 row: row,
183                 column: column
184             });
185         }
186         //左上角
187         row = p.row - i;
188         column = p.column + i;
189         if (this.isValid(row, column)) {
190             points.push({
191                 row: row,
192                 column: column
193             });
194         }
195         //
196         row = p.row;
197         column = p.column - i;
198         if (this.isValid(row, column)) {
199             points.push({
200                 row: row,
201                 column: column
202             });
203         }
204         //
205         row = p.row - i;
206         column = p.column;
207         if (this.isValid(row, column)) {
208             points.push({
209                 row: row,
210                 column: column
211             });
212         }
213     }
214     return points;
215 };
216 //该位置是否合法
217 Chessboard.prototype.isValid = function(row, column) {
218     return row >= 0 && row < this.row && column >= 0 && column < this.column && this.data[row][column] == Chessboard.NONE;
219 };
220 /**
221  * [availableSteps 获取可走的位置]
222  * @return {[type]} [description]
223  */
224 Chessboard.prototype.availableSteps = function() {
225     var availableSteps = [],
226         row = this.row,
227         column = this.column,
228         stackLen = this.stack.length,
229         centerRow = Math.floor((row - 1) / 2),
230         centerColumn = Math.floor((column - 1) / 2);
231     if (!stackLen || (stackLen == 1 && this.data[centerRow][centerColumn] == Chessboard.NONE)) {
232         availableSteps.push({
233             row: centerRow,
234             column: centerColumn
235         });
236         return availableSteps;
237     } else {
238         if (stackLen == 1) {
239             var nextRow = centerRow + (Math.random() < 0.5 ? -1 : 1),
240                 nextColumn = centerColumn + (Math.random() < 0.5 ? -1 : 1);
241             availableSteps.push({
242                 row: nextRow,
243                 column: nextColumn
244             });
245             return availableSteps;
246         } else {
247             var hash = {};
248             this.stack.forEach(function(p) {
249                 nearPoints = this.getNearPoints(p);
250                 nearPoints.forEach(function(nearPoint) {
251                     var row = nearPoint.row,
252                         column = nearPoint.column;
253                     if (!hash[row + '#' + column]) {
254                         availableSteps.push(nearPoint);
255                         hash[row + '#' + column] = true;
256                     }
257                 });
258             }.bind(this));
259             return availableSteps;
260         }
261     }
262     return availableSteps;
263 };
264 Chessboard.prototype.analyseMax = function(data, type) {
265     switch (type) {
266         case Chessboard.FIVE_TYPE:
267             return ~data.indexOf('11111') ? 1 : 0;
268         case Chessboard.SFOUR_TYPE:
269             if (~data.indexOf('011110')) {
270                 return 1;
271             }
272             return 0;
273         case Chessboard.FOUR_TYPE:
274             var c = 0;
275             var res1 = data.match(/211110/g);
276             var res2 = data.match(/011112/g);
277             var res3 = data.match(/10111/g);
278             var res4 = data.match(/11011/g);
279             var res5 = data.match(/11101/g);
280             c += (res1 ? res1.length : 0);
281             c += (res2 ? res2.length : 0);
282             c += (res3 ? res3.length : 0);
283             c += (res4 ? res4.length : 0);
284             c += (res5 ? res5.length : 0);
285             return c;
286         case Chessboard.STHREE_TYPE:
287             var c = 0;
288             var res1 = data.match(/01110/g);
289             var res2 = data.match(/011010/g);
290             var res3 = data.match(/010110/g);
291             c += (res1 ? res1.length : 0);
292             c += (res2 ? res2.length : 0);
293             c += (res3 ? res3.length : 0);
294             return c;
295         case Chessboard.THREE_TYPE:
296             var c = 0;
297             var res1 = data.match(/211100/g);
298             var res2 = data.match(/001112/g);
299             var res3 = data.match(/211010/g);
300             var res4 = data.match(/010112/g);
301             var res5 = data.match(/210110/g);
302             var res6 = data.match(/011012/g);
303             var res7 = data.match(/10011/g);
304             var res8 = data.match(/11001/g);
305             var res9 = data.match(/10101/g);
306             var res10 = data.match(/2011102/g);
307             c += (res1 ? res1.length : 0);
308             c += (res2 ? res2.length : 0);
309             c += (res3 ? res3.length : 0);
310             c += (res4 ? res4.length : 0);
311             c += (res5 ? res5.length : 0);
312             c += (res6 ? res6.length : 0);
313             c += (res7 ? res7.length : 0);
314             c += (res8 ? res8.length : 0);
315             c += (res9 ? res9.length : 0);
316             c += (res10 ? res10.length : 0);
317             return c;
318         case Chessboard.STWO_TYPE:
319             var c = 0;
320             var res1 = data.match(/001100/g);
321             var res2 = data.match(/01010/g);
322             var res3 = data.match(/010010/g);
323             c += (res1 ? res1.length : 0);
324             c += (res2 ? res2.length : 0);
325             c += (res3 ? res3.length : 0);
326             return c;
327         case Chessboard.TWO_TYPE:
328             var c = 0;
329             var res1 = data.match(/001100/g);
330             var res2 = data.match(/01010/g);
331             var res3 = data.match(/010010/g);
332             c += (res1 ? res1.length : 0);
333             c += (res2 ? res2.length : 0);
334             c += (res3 ? res3.length : 0);
335             return c;
336         default:
337             return 0;
338     }
339 };
340 Chessboard.prototype.analyseMin = function(data, type) {
341     switch (type) {
342         case Chessboard.FIVE_TYPE:
343             return ~data.indexOf('22222') ? 1 : 0;
344         case Chessboard.SFOUR_TYPE:
345             if (~data.indexOf('022220')) {
346                 return 1;
347             }
348             return 0;
349         case Chessboard.FOUR_TYPE:
350             var c = 0;
351             var res1 = data.match(/122220/g);
352             var res2 = data.match(/022221/g);
353             var res3 = data.match(/20222/g);
354             var res4 = data.match(/22022/g);
355             var res5 = data.match(/22202/g);
356             c += (res1 ? res1.length : 0);
357             c += (res2 ? res2.length : 0);
358             c += (res3 ? res3.length : 0);
359             c += (res4 ? res4.length : 0);
360             c += (res5 ? res5.length : 0);
361             return c;
362         case Chessboard.STHREE_TYPE:
363             var c = 0;
364             var res1 = data.match(/02220/g);
365             var res2 = data.match(/022020/g);
366             var res3 = data.match(/020220/g);
367             c += (res1 ? res1.length : 0);
368             c += (res2 ? res2.length : 0);
369             c += (res3 ? res3.length : 0);
370             return c;
371         case Chessboard.THREE_TYPE:
372             var c = 0;
373             var res1 = data.match(/122200/g);
374             var res2 = data.match(/002221/g);
375             var res3 = data.match(/122020/g);
376             var res4 = data.match(/020221/g);
377             var res5 = data.match(/120220/g);
378             var res6 = data.match(/022021/g);
379             var res7 = data.match(/20022/g);
380             var res8 = data.match(/22002/g);
381             var res9 = data.match(/20202/g);
382             var res10 = data.match(/1022201/g);
383             c += (res1 ? res1.length : 0);
384             c += (res2 ? res2.length : 0);
385             c += (res3 ? res3.length : 0);
386             c += (res4 ? res4.length : 0);
387             c += (res5 ? res5.length : 0);
388             c += (res6 ? res6.length : 0);
389             c += (res7 ? res7.length : 0);
390             c += (res8 ? res8.length : 0);
391             c += (res9 ? res9.length : 0);
392             c += (res10 ? res10.length : 0);
393             return c;
394         case Chessboard.STWO_TYPE:
395             var c = 0;
396             var res1 = data.match(/002200/g);
397             var res2 = data.match(/02020/g);
398             var res3 = data.match(/020020/g);
399             c += (res1 ? res1.length : 0);
400             c += (res2 ? res2.length : 0);
401             c += (res3 ? res3.length : 0);
402             return c;
403         case Chessboard.TWO_TYPE:
404             var c = 0;
405             var res1 = data.match(/001100/g);
406             var res2 = data.match(/01010/g);
407             var res3 = data.match(/010010/g);
408             c += (res1 ? res1.length : 0);
409             c += (res2 ? res2.length : 0);
410             c += (res3 ? res3.length : 0);
411             return c;
412         default:
413             return 0;
414     }
415 };
416 /**
417  * [evaluate 计算当前棋盘的估值]
418  * @return {[type]} [description]
419  */
420 Chessboard.prototype.evaluate = function() {
421     var maxW = minW = 0;
422     var maxGroup = {
423             "5": 0,
424             "4": 0,
425             "3": 0,
426             "2": 0,
427             "1": 0
428         },
429         minGroup = {
430             "5": 0,
431             "4": 0,
432             "3": 0,
433             "2": 0,
434             "1": 0
435         };
436     for (var i = 0; i < this.count; i++) {
437         if (this.maxWin[i].max == 5 && !this.maxWin[i].min) {
438             return Chessboard.MAX_VALUE;
439         }
440         if (this.minWin[i].min == 5 && !this.minWin[i].max) {
441             return Chessboard.MIN_VALUE;
442         }
443         if (this.maxWin[i].max == 4 && !this.maxWin[i].min) {
444             maxGroup[4]++;
445         }
446         if (this.minWin[i].min == 4 && !this.minWin[i].max) {
447             minGroup[4]++;
448         }
449         if (this.maxWin[i].max == 3 && !this.maxWin[i].min) {
450             maxGroup[3]++;
451         }
452         if (this.minWin[i].min == 3 && !this.minWin[i].max) {
453             minGroup[3]++;
454         }
455         if (this.maxWin[i].max == 2 && !this.maxWin[i].min) {
456             maxGroup[2]++;
457         }
458         if (this.minWin[i].min == 2 && !this.minWin[i].max) {
459             minGroup[2]++;
460         }
461         if (this.maxWin[i].max == 1 && !this.maxWin[i].min) {
462             maxGroup[1]++;
463         }
464         if (this.minWin[i].min == 1 && !this.minWin[i].max) {
465             minGroup[1]++;
466         }
467     }
468     maxW = maxGroup[4] * Chessboard.FOUR_W + maxGroup[3] * Chessboard.THREE_W + maxGroup[2] * Chessboard.TWO_W + maxGroup[1] * Chessboard.ONE_W;
469     minW = minGroup[4] * Chessboard.FOUR_W + minGroup[3] * Chessboard.THREE_W + minGroup[2] * Chessboard.TWO_W + minGroup[1] * Chessboard.ONE_W;
470     return maxW - minW;
471 };
472 /**
473  * [isMaxWin 人是否赢了]
474  * @return {Boolean} [description]
475  */
476 Chessboard.prototype.isMaxWin = function() {
477     var w = this.evaluate();
478     return w == Chessboard.MAX_VALUE ? true : false;
479 };
480 /**
481  * [isMinWin AI是否赢了]
482  * @return {Boolean} [description]
483  */
484 Chessboard.prototype.isMinWin = function() {
485     var w = this.evaluate();
486     return w == Chessboard.MIN_VALUE ? true : false;
487 };
488 /**
489  * [end 结束游戏]
490  * @return {[type]} [description]
491  */
492 Chessboard.prototype.end = function() {
493     this.is_ended = true;
494     return this;
495 };
496 /**
497  * [isEnded 游戏是否结束]
498  * @return {Boolean} [description]
499  */
500 Chessboard.prototype.isEnded = function() {
501     return this.is_ended;
502 };
503 /**
504  * [max max下棋]
505  * @param  {[type]} currentChessboard [当前棋盘]
506  * @param  {[type]} depth        [考虑深度]
507  * @return {[type]}              [description]
508  */
509 var max = function(currentChessboard, depth, beta) {
510     //记录优势值,应该下棋的位置
511     var row, column, alpha = -Infinity;
512     //什么都不下,直接返回当前棋盘评估值
513     if (depth == 0) {
514         alpha = currentChessboard.evaluate();
515         return {
516             w: alpha
517         };
518     } else {
519         //获取每一步可以走的方案
520         var steps = currentChessboard.availableSteps();
521         // console.log('搜索MAX' + steps.length + '个棋局');
522         if (steps.length) {
523             //对于每一种走法
524             for (var i = 0, l = steps.length; i < l; i++) {
525                 var step = steps[i];
526                 //下棋
527                 currentChessboard.put(step.row, step.column, Chessboard.MAX);
528                 //如果已经赢了,则直接下棋,不再考虑对方下棋
529                 if (currentChessboard.isMaxWin()) {
530                     alpha = Chessboard.MAX_VALUE;
531                     row = step.row;
532                     column = step.column;
533                     //退回上一步下棋
534                     currentChessboard.rollback();
535                     break;
536                 } else {
537                     //考虑对方depth-1步下棋之后的优势值,如果对方没棋可下了,则返回当前棋盘估值
538                     var res = min(currentChessboard, depth - 1) || {
539                         w: currentChessboard.evaluate()
540                     };
541                     //退回上一步下棋
542                     currentChessboard.rollback();
543                     if (res.w > alpha) {
544                         //选择最大优势的走法
545                         alpha = res.w;
546                         row = step.row;
547                         column = step.column;
548                     }
549                     //如果人可以获得更好的走法,则AI必然不会选择这一步走法,所以不用再考虑人的其他走法
550                     if (alpha >= beta) {
551                         // console.log('MAX节点' + l + '个棋局,剪掉了' + (l - 1 - i) + '个MIN棋局');
552                         break;
553                     }
554                 }
555 
556             }
557             return {
558                 w: alpha,
559                 row: row,
560                 column: column
561             };
562         }
563     }
564 };
565 /**
566  * [min min下棋]
567  * @param  {[type]} currentChessboard [当前棋盘]
568  * @param  {[type]} depth        [考虑深度]
569  * @return {[type]}              [权重和当前推荐下棋的位置]
570  */
571 var min = function(currentChessboard, depth, alpha) {
572     var row, column, beta = Infinity;
573     if (depth == 0) {
574         beta = currentChessboard.evaluate();
575         return {
576             w: beta
577         };
578     } else {
579         //获取每一步可以走的方案
580         var steps = currentChessboard.availableSteps();
581         // console.log('搜索MIN' + steps.length + '个棋局');
582         if (steps.length) {
583             //对于每一种走法
584             for (var i = 0, l = steps.length; i < l; i++) {
585                 var step = steps[i];
586                 //下棋
587                 currentChessboard.put(step.row, step.column, Chessboard.MIN);
588                 //如果已经赢了,则直接下棋,不再考虑对方下棋
589                 if (currentChessboard.isMinWin()) {
590                     beta = Chessboard.MIN_VALUE;
591                     row = step.row;
592                     column = step.column;
593                     //退回上一步下棋
594                     currentChessboard.rollback();
595                     break;
596                 } else {
597                     //考虑对方depth-1步下棋之后的优势值,如果对方没棋可下了,则返回当前棋盘估值
598                     var res = max(currentChessboard, depth - 1, beta) || {
599                         w: currentChessboard.evaluate()
600                     };
601                     //退回上一步下棋
602                     currentChessboard.rollback();
603                     if (res.w < beta) {
604                         //选择最大优势的走法
605                         beta = res.w;
606                         row = step.row;
607                         column = step.column;
608                     }
609                     //如果AI可以获得更好的走法,则人必然不会选择这一步走法,所以不用再考虑AI的其他走法
610                     if (beta <= alpha) {
611                         // console.log('MIN节点' + l + '个棋局,剪掉了' + (l - 1 - i) + '个MAX棋局');
612                         break;
613                     }
614                 }
615             }
616             return {
617                 w: beta,
618                 row: row,
619                 column: column
620             };
621         }
622     }
623 };
624 Chessboard.NONE = 0;
625 Chessboard.MAX = 1;
626 Chessboard.MIN = 2;
627 Chessboard.FIVE_TYPE = 1;
628 Chessboard.SFOUR_TYPE = 2;
629 Chessboard.FOUR_TYPE = 3;
630 Chessboard.STHREE_TYPE = 4;
631 Chessboard.THREE_TYPE = 5;
632 Chessboard.STWO_TYPE = 6;
633 Chessboard.TWO_TYPE = 7;
634 Chessboard.MAX_VALUE = 100000;
635 Chessboard.MIN_VALUE = -100000;
636 Chessboard.FIVE_W = 100000;
637 Chessboard.SFOUR_W = 10000;
638 Chessboard.FOUR_W = 5000;
639 Chessboard.STHREE_W = 2000;
640 Chessboard.THREE_W = 1000;
641 Chessboard.STWO_W = 500;
642 Chessboard.TWO_W = 200;
643 Chessboard.ONE_W = 10;

 

posted @ 2023-02-19 16:28  来杯明前奶绿  阅读(164)  评论(0编辑  收藏  举报