Làm thế nào tôi có thể chuyển đổi các bước di chuyển có thể có của một quân cờ thành một bước di chuyển thực tế trong bitboard?
Nếu tôi có một bitboard (ví dụ: đối với hình vuông hiệp sĩ anh ta có thể di chuyển đến) như thế này:
00000000
00000000
00000000
00000000
00000000
01000000
00100000
N0000000
Làm cách nào để tôi có thể "chuyển đổi" nó thành một nước đi hoặc "áp dụng" nước đi đó?
Trả lời
Thực sự có hai câu hỏi trong bài đăng của bạn
Làm thế nào để chuyển đổi các đòn tấn công trong Bitboards thành một danh sách di chuyển?
Làm thế nào để biểu diễn các nước đi đó?
- Làm thế nào để biểu diễn các nước đi đó?
Câu hỏi này đã được thảo luận ở đây và ở đây kỹ lưỡng. Nhưng để cung cấp một số ngữ cảnh cho người đọc của bài đăng này, một cách rất đơn giản để làm điều này có thể là sử dụng một lớp / cấu trúc để đại diện cho một động thái
- Từ (vị trí của mảnh chuyển động)
- Tới (vị trí của mảnh chuyển động)
- Cờ di chuyển đặc biệt (cho dù nước đi là một nước đi đặc biệt, tức là vượt qua, sử dụng, thăng cấp)
- Mảnh quảng cáo (nếu nước đi là một động thái thăng hạng, thì quân cờ đó đã quảng bá cho cái gì)
Bạn cũng có thể có các thuộc tính bổ sung như mảnh chiếm được, loại lâu đài, v.v. Nhưng thông tin đó cũng có thể được suy ra bằng cách sử dụng From
và To
hình vuông.
Vì vậy, xem xét ở trên, chúng ta có thể biểu diễn động thái như sau
struct Move {
char from;
char to;
char spMoveFlag;
char promotedPiece;
};
Tuy nhiên, rất nhiều công cụ cờ vua, bao gồm cả công cụ của tôi sử dụng một phương pháp khác có thể nén nước đi thành 16
từng bit. Đọc thêm về điều đó ở đây .
Khi chúng ta biết động thái là gì, chúng ta có thể trả lời câu hỏi tiếp theo
- Làm thế nào để chuyển đổi các đòn tấn công trong Bitboards thành một danh sách di chuyển?
Hãy coi rằng chúng ta có một hiệp sĩ trên quảng trường d4 . Chúng tôi cũng có một chức năng trả về các cuộc tấn công của một hiệp sĩ cho hình vuông.
. . . . . . . .
. . . . . . . .
. . 1 . 1 . . .
. 1 . . . 1 . .
. . . n . . . . = knightattacks(SQ_D4)
. 1 . . . 1 . .
. . 1 . 1 . . .
. . . . . . . .
Vì vậy, hãy lưu trữ điều này bên trong một biến knightAttacks
uint64_t knightAttacks = getKnightAttacksBB(SQ_D4);
Nếu bạn để ý kỹ, chúng ta đã có 1 trong 4 thuộc tính chúng ta cần để đại diện cho nước đi, đó là From
hình vuông -D4
. Bây giờ tất cả những gì chúng ta cần làm là bằng cách nào đó có được To
hình vuông để hoàn thành việc này vì một hiệp sĩ không thể thực hiện bất kỳ loại di chuyển đặc biệt nào.
Một phương pháp rất phổ biến là bật bit ít quan trọng nhất cho đến khi bạn knightAttacks
không còn gì cả 0
.
0000011100001000
^
Least significant bit (lsb)
Bạn có thể theo liên kết mà tôi đã cung cấp để biết cách bạn có thể thực hiện việc này hoặc có khả năng trình biên dịch của bạn có thể đã cung cấp cho bạn.
Bạn chỉ cần tạo một hàm xóa lsb của bitboard và trả về chỉ mục của nó. Hãy gọi nópoplsb()
.
Nhìn chung, mã của bạn có thể trông như thế này
int from = SQ_D4;
uint16_t knightAttacks = getknightAttacks(from);
while (knightAttacks) {
int to = poplsb(knightAttacks);
MyMoveList.Add( NewMove(from, to, 0, 0) ); // The last two 0's denote that there is no special move
}
Điều này hoạt động bởi vì mỗi lần bạn bật lsb, một bit sẽ bị xóa cho đến khi số là == 0. Đó là khi vòng lặp dừng lại và bạn có một danh sách hoàn toàn tốt các nước đi. Ý tưởng tương tự cũng áp dụng cho bất kỳ loại di chuyển nào bạn đã tạo bằng Bitboards.