Nearly forty years ago, a friend of mine challenged me to write a program to solve The Knight's Tour problem. Simply put: can a knight move around the chess board landing on every space only once?
At the time, I was working with a WYSE-386 computer with 2MB of memory and 40MB of hard drive space. After coming up with a suitable algorithm, I implemented it in 16-bit assembler. Running on a single CPU at 16MHz, It took 29 days for it to find a solution.
Since then, I have periodically revisited this using a number of different approaches. The most recent iteration of my algorithm took another turn. Instead of using a two-dimensional array for the board, I linearized it into a single vector which provides substantial computational advantages over my previous efforts.
I start by defining the board to virtually look like this:
These moves are stored in an eight-element array of offsets ("dir[8]") from the current location. Given the direction (0-7), adding the value of the ith element gives the new location. If the new location is not zero, then it is an invalid move.
When 63 moves have been made, the program stops and prints out the board. Here is the basic C code to make it happen:
// knight.c (Knight's Tour)
#include <stdio.h> // test
#include <stdbool.h>
int dir[] = { -25,-14,10,23,25,14,-10,-23 };
void print_board(int board[144])
{
  for(int i=110;i>25;i-=12)
  {
    for(int j=0;j<8;j++)
    {
      printf("\t%d",board[i+j]);
    }
    printf("\n\n");
  }
}
bool find_next(int curloc, int moves, int board[144])
{
  if(moves > 64)
  {
    return true;
  }
  for(int i = 0; i < 8; i++)
  {
    int tstloc = curloc + dir[i];
    if(board[tstloc] == 0)
    {
      board[tstloc] = moves;
      if(find_next(tstloc, moves + 1, board))
      {
        return true;
      }
      else
      {
        board[tstloc] = 0;
      }
    }
  }
  return false;
}
int main(void)
{
  int i,j;
  int board[144];
  for (i = 0; i < 144; i++)
  {
    board[i] = 1;
  }
  for (i = 2; i < 10; i++)
  {
    for (j = 2; j < 10; j++)
    {
      board[i + j * 12] = 0;
    }
  }
  board[32] = 1;
  if (find_next(32, 2, board))
  {
    printf("Knight's Tour solution:\n");
    print_board(board);
  }
  else
  {
    printf("No solution found.\n");
  }
  return 0;
}
#include <stdio.h>
#include <stdbool.h>
#define N 8 // Size of the chessboard
int moveX[8] = {2, 1, -1, -2, -2, -1, 1, 2};
int moveY[8] = {1, 2, 2, 1, -1, -2, -2, -1};
bool isSafe(int x, int y, int sol[N][N]) {
    return (x >= 0 && x < N && y >= 0 && y < N && sol[x][y] == -1);
}
void printSolution(int sol[N][N]) {
    for (int x = 0; x < N; x++) {
        for (int y = 0; y < N; y++) {
            printf("%2d ", sol[x][y]);
        }
        printf("\n");
    }
}
bool solveKnightTour(int x, int y, int moveCount, int sol[N][N]) {
    if (moveCount == N * N) {
        return true; // All squares visited
    }
    for (int i = 0; i < 8; i++) {
        int nextX = x + moveX[i];
        int nextY = y + moveY[i];
        if (isSafe(nextX, nextY, sol)) {
            sol[nextX][nextY] = moveCount;
            if (solveKnightTour(nextX, nextY, moveCount + 1, sol)) {
                return true; // Move leads to a solution
            } else {
                sol[nextX][nextY] = -1; // Backtrack
            }
        }
    }
    return false; // No solution found
}
int main() {
    int sol[N][N]; // Solution matrix
    // Initialize solution matrix
    for (int x = 0; x < N; x++) {
        for (int y = 0; y < N; y++) {
            sol[x][y] = -1; // Mark all squares as unvisited
        }
    }
    // Starting position
    int startX = 0;
    int startY = 0;
    sol[startX][startY] = 0; // Mark starting position as visited
    if (solveKnightTour(startX, startY, 1, sol)) {
        printf("Knight's Tour solution:\n");
        printSolution(sol);
    } else {
        printf("No solution found.\n");
    }
    return 0;
}



 
 
No comments:
Post a Comment