upgrade ai

This commit is contained in:
pengfei.zhou 2019-12-17 16:39:59 +08:00
parent d7a95b5958
commit 2850f2a436

View File

@ -11,15 +11,19 @@ const count = 13
class AIComputer { class AIComputer {
wins: Array<Array<Array<Boolean>>> = new Array(count).fill(0).map(_ => new Array(count).fill(0).map(_ => [])) wins: Array<Array<{ x: number, y: number }>> = []
winCount = 0 winCount = 0
blackWins: number[] matrix: Map<number, State>
whiteWins: number[] constructor(matrix: Map<number, State>) {
constructor() { this.matrix = matrix
for (let y = 0; y < count; y++) { for (let y = 0; y < count; y++) {
for (let x = 0; x < count - 4; x++) { for (let x = 0; x < count - 4; x++) {
this.wins.push([])
for (let k = 0; k < 5; k++) { for (let k = 0; k < 5; k++) {
this.wins[x + k][y][this.winCount] = true; this.wins[this.winCount].push({
x: x + k,
y,
})
} }
this.winCount++; this.winCount++;
} }
@ -27,8 +31,12 @@ class AIComputer {
for (let x = 0; x < count; x++) { for (let x = 0; x < count; x++) {
for (let y = 0; y < count - 4; y++) { for (let y = 0; y < count - 4; y++) {
this.wins.push([])
for (let k = 0; k < 5; k++) { for (let k = 0; k < 5; k++) {
this.wins[x][y + k][this.winCount] = true; this.wins[this.winCount].push({
x,
y: y + k,
})
} }
this.winCount++; this.winCount++;
} }
@ -36,8 +44,12 @@ class AIComputer {
for (let x = 0; x < count - 4; x++) { for (let x = 0; x < count - 4; x++) {
for (let y = 0; y < count - 4; y++) { for (let y = 0; y < count - 4; y++) {
this.wins.push([])
for (let k = 0; k < 5; k++) { for (let k = 0; k < 5; k++) {
this.wins[x + k][y + k][this.winCount] = true; this.wins[this.winCount].push({
x: x + k,
y: y + k,
})
} }
this.winCount++; this.winCount++;
} }
@ -45,28 +57,16 @@ class AIComputer {
for (let x = 0; x < count - 4; x++) { for (let x = 0; x < count - 4; x++) {
for (let y = count - 1; y > 3; y--) { for (let y = count - 1; y > 3; y--) {
this.wins.push([])
for (let k = 0; k < 5; k++) { for (let k = 0; k < 5; k++) {
this.wins[x + k][y - k][this.winCount] = true; this.wins[this.winCount].push({
x: x + k,
y: y - k,
})
} }
this.winCount++; this.winCount++;
} }
} }
this.blackWins = new Array(this.winCount).fill(0)
this.whiteWins = new Array(this.winCount).fill(0)
}
oneStep(idx: number, role: State.BLACK | State.WHITE) {
const { x, y } = this.index2Position(idx)
for (let loop = 0; loop < this.winCount; loop++) {
if (this.wins[x][y][loop]) {
if (role === State.BLACK) {
this.blackWins[loop] += 1
} else {
this.whiteWins[loop] += 1
}
}
}
} }
index2Position(idx: number) { index2Position(idx: number) {
@ -74,6 +74,41 @@ class AIComputer {
const y = Math.floor(idx / count) const y = Math.floor(idx / count)
return { x, y } return { x, y }
} }
get blackWins() {
return this.wins.map((win) => {
let idx = 0
for (let e of win) {
switch (this.matrix.get(e.x + e.y * count)) {
case State.BLACK:
idx++
break
case State.WHITE:
return 0
default:
break
}
}
return idx
})
}
get whiteWins() {
return this.wins.map((win) => {
let count = 0
for (let e of win) {
switch (this.matrix.get(e.x + e.y * count)) {
case State.WHITE:
count++
break
case State.BLACK:
return 0
default:
break
}
}
return count
})
}
compute(matrix: State[], role: State.BLACK | State.WHITE) { compute(matrix: State[], role: State.BLACK | State.WHITE) {
const myScore = new Array(matrix.length).fill(0) const myScore = new Array(matrix.length).fill(0)
@ -86,10 +121,11 @@ class AIComputer {
if (state != State.Unspecified) { if (state != State.Unspecified) {
return return
} }
const { x, y } = this.index2Position(idx) this.wins.forEach((e, winIdx) => {
for (let loop = 0; loop < this.winCount; loop++) { if (e.filter(e => (e.x + e.y * count) === idx).length === 0) {
if (this.wins[x][y][loop]) { return
switch (rivalWins[loop]) { }
switch (rivalWins[winIdx]) {
case 1: case 1:
rivalScore[idx] += 200 rivalScore[idx] += 200
break break
@ -106,7 +142,7 @@ class AIComputer {
break break
} }
switch (myWins[loop]) { switch (myWins[winIdx]) {
case 1: case 1:
myScore[idx] += 220 myScore[idx] += 220
break break
@ -122,8 +158,7 @@ class AIComputer {
default: default:
break break
} }
} })
}
if (rivalScore[idx] > max) { if (rivalScore[idx] > max) {
max = rivalScore[idx]; max = rivalScore[idx];
retIdx = idx retIdx = idx
@ -141,8 +176,6 @@ class AIComputer {
retIdx = idx retIdx = idx
} }
} }
}) })
return retIdx return retIdx
} }
@ -307,15 +340,9 @@ class GoBangVM extends ViewModel<GoBangState, GoBangVH>{
if (it.role === 'black') { if (it.role === 'black') {
it.matrix.set(idx, State.BLACK) it.matrix.set(idx, State.BLACK)
it.role = 'white' it.role = 'white'
if (this.computer) {
this.computer.oneStep(idx, State.BLACK)
}
} else { } else {
it.matrix.set(idx, State.WHITE) it.matrix.set(idx, State.WHITE)
it.role = 'black' it.role = 'black'
if (this.computer) {
this.computer.oneStep(idx, State.WHITE)
}
} }
it.anchor = undefined it.anchor = undefined
if (this.checkResult(idx)) { if (this.checkResult(idx)) {
@ -329,9 +356,13 @@ class GoBangVM extends ViewModel<GoBangState, GoBangVH>{
}) })
} else { } else {
if (it.role === 'black' && it.gameMode === GameMode.C2P) { if (it.role === 'black' && it.gameMode === GameMode.C2P) {
setTimeout(() => {
this.computeNextStep(it) this.computeNextStep(it)
}, 0)
} else if (it.role === 'white' && it.gameMode === GameMode.P2C) { } else if (it.role === 'white' && it.gameMode === GameMode.P2C) {
setTimeout(() => {
this.computeNextStep(it) this.computeNextStep(it)
}, 0)
} }
} }
}) })
@ -388,7 +419,6 @@ class GoBangVM extends ViewModel<GoBangState, GoBangVH>{
do { do {
idx = this.computer.compute(tempMatrix, it.role === 'black' ? State.BLACK : State.WHITE) idx = this.computer.compute(tempMatrix, it.role === 'black' ? State.BLACK : State.WHITE)
} while (it.matrix.get(idx) === State.Unspecified) } while (it.matrix.get(idx) === State.Unspecified)
this.computer.oneStep(idx, it.role === 'black' ? State.BLACK : State.WHITE)
this.updateState(state => { this.updateState(state => {
state.matrix.set(idx, state.role === 'black' ? State.BLACK : State.WHITE) state.matrix.set(idx, state.role === 'black' ? State.BLACK : State.WHITE)
state.role = state.role === 'black' ? 'white' : 'black' state.role = state.role === 'black' ? 'white' : 'black'
@ -409,10 +439,9 @@ class GoBangVM extends ViewModel<GoBangState, GoBangVH>{
it.matrix.clear() it.matrix.clear()
it.role = "black" it.role = "black"
it.anchor = undefined it.anchor = undefined
this.computer = new AIComputer this.computer = new AIComputer(it.matrix)
if (it.gameMode === GameMode.C2P) { if (it.gameMode === GameMode.C2P) {
const idx = Math.floor(Math.random() * count) * count + Math.floor(Math.random() * count) const idx = Math.floor(Math.random() * count) * count + Math.floor(Math.random() * count)
this.computer.oneStep(idx, State.BLACK)
it.matrix.set(idx, State.BLACK) it.matrix.set(idx, State.BLACK)
it.role = 'white' it.role = 'white'
} }