-
马踏棋盘问题(C语言)——基于贪心算法优化的深度优先搜索
-
(动态展示马的路径)
-
附带蹩马脚功能,将main函数中 //set_barrier(3, 4); //set_barrier(6,
注释取消,即可在棋盘3行4列与6行4列的位置增加不可走区域,以此类推 -
如果有人看,我再写详细一些
#include<stdio.h>
#include<stdlib.h>
#include<time.h>
#include<Windows.h>
#define WIDTH 24
#define VERTICAL 1 //垂直的路
#define HORIZONTAL 0 //水平的路
#define Horse_Tag 7
#define LEFT_2_UP_1 'A'
#define LEFT_1_UP_2 'B'
#define RIGHT_1_UP_2 'C'
#define RIGHT_2_UP_1 'D'
#define RIGHT_2_DOWN_1 'E'
#define RIGHT_1_DOWN_2 'F'
#define LEFT_1_DOWN_2 'G'
#define LEFT_2_DOWN_1 'H'
#define OVER 'O';
#define WALL 1
typedef struct position {
int value; // '0' represents a wall, while '1' represents an accessible position.
bool isvisited; //马是否来过这个位置了
}PST;
typedef struct node
{
int x;
int y;
}NODE;
typedef struct ar
{
int num;
char togo;
}ARRAY;
PST matrix[WIDTH + 2][WIDTH + 2];
NODE path[WIDTH*WIDTH];
int found = 0; //用于记录找到的路径个数
int count_barrier = 0; //用于记录设置的障碍个数
void init()
{
void set_road(int col, int value);
for (int i = 0; i < WIDTH + 1; i++)
{
for (int j = 0; j < WIDTH + 1; j++)
{
matrix[i][j].value = 0;
matrix[i][j].isvisited = false;
}
}
set_road(0, VERTICAL); //设置地图的边框
set_road(0, HORIZONTAL);
set_road(WIDTH + 1, HORIZONTAL);
set_road(WIDTH + 1, VERTICAL);
}
void set_road(int col, int value) //col代表将哪一行或者列设置成路(将他们的值++); if the value equals 1,it means the road would be set vertically, otherwise horizontally.
{
if (value == VERTICAL)
{
for (int i = 0; i < WIDTH + 1; i++)
{
matrix[col][i].value++;
}
}
else
{
for (int i = 0; i < WIDTH + 1; i++)
{
matrix[i][col].value++;
}
}
}
void set_horse_position(int &horse_x, int &horse_y, int x, int y) //To preset the horse's position at the very begining
{
matrix[x][y].value = Horse_Tag;
horse_x = x;
horse_y = y;
}
void set_barrier(int x, int y) //设置障碍,别马脚
{
matrix[x][y].value = 1;
matrix[x][y].isvisited = true;
count_barrier++;
}
void gotoxy(unsigned char x, unsigned char y) {
//COORD是Windows API中定义的一种结构,表示一个字符在控制台屏幕上的坐标
COORD cor;
//句柄
HANDLE hout;
//设定我们要定位到的坐标
cor.X = 2 * y;
cor.Y = x;
//GetStdHandle函数获取一个指向特定标准设备的句柄,包括标准输入,标准输出和标准错误。
//STD_OUTPUT_HANDLE正是代表标准输出(也就是显示屏)的宏
hout = GetStdHandle(STD_OUTPUT_HANDLE);
//SetConsoleCursorPosition函数用于设置控制台光标的位置
SetConsoleCursorPosition(hout, cor);
}
void show_path()
{
system("cls");
for (int i = 0; i < WIDTH + 2; i++)
{
for (int j = 0; j < WIDTH + 2; j++)
{
if (matrix[i][j].value != WALL)
printf(" ", matrix[i][j].value);
else
printf("* ");
}
printf("\n");
}
printf("\n\n开始展示找到的第%d条路径", found); //地图生成完毕
for (int i = 0; i < WIDTH*WIDTH; i++) //将路径动态显示出来
{
gotoxy(path[i].x, path[i].y);
printf("M");
Sleep(50);
}
}
int get_next(int x, int y, ARRAY next_step[], int step, int count) //用于寻找马周围八个方向中接下来一步可走位置最少的一个位置
{
int k = 0;
if (x - 1 > 0 && y - 2 > 0 && matrix[x - 1][y - 2].isvisited == false && matrix[x][y - 1].value == 0) // 左2上1 可以走吗? 并且检测是否有别马脚
{
k++;
if (count < 1)
{
next_step[0].num = get_next(x - 1, y - 2, next_step, step, count + 1);
next_step[0].togo = LEFT_2_UP_1;
if (step == WIDTH * WIDTH - 1 - count_barrier) //如果走到了最后一步,还只剩下最后一个格子的情况
{
next_step[0].num = 1; //将值置为一个正数(不一定要是1),以便于最后一步可执行 。否则最后一步会检测到无路可走
}
}
}
if (x - 2 > 0 && y - 1 > 0 && matrix[x - 2][y - 1].isvisited == false && matrix[x - 1][y].value == 0) // 左1上2可走吗? 并且检测是否有别马脚
{
k++;
if (count < 1)
{
next_step[1].num = get_next(x - 2, y - 1, next_step, step, count + 1);
next_step[1].togo = LEFT_1_UP_2;
if (step == WIDTH * WIDTH - 1 - count_barrier) //如果走到了最后一步,还只剩下最后一个格子的情况
{
next_step[1].num = 1; //将值置为一个正数(不一定要是1),以便于最后一步可执行 。否则最后一步会检测到无路可走
}
}
}
if (x - 2 > 0 && y + 1 <= WIDTH && matrix[x - 2][y + 1].isvisited == false && matrix[x - 1][y].value == 0) // 右1上2可以走吗
{
k++;
if (count < 1)
{
next_step[2].num = get_next(x - 2, y + 1, next_step, step, count + 1);
next_step[2].togo = RIGHT_1_UP_2;
if (step == WIDTH * WIDTH - 1 - count_barrier) //如果走到了最后一步,还只剩下最后一个格子的情况
{
next_step[2].num = 1; //将值置为一个正数(不一定要是1),以便于最后一步可执行 。否则最后一步会检测到无路可走
}
}
}
if (x - 1 > 0 && y + 2 <= WIDTH && matrix[x - 1][y + 2].isvisited == false && matrix[x][y + 1].value == 0) // 右2上1可以走吗
{
k++;
if (count < 1)
{
next_step[3].num = get_next(x - 1, y + 2, next_step, step, count + 1);
next_step[3].togo = RIGHT_2_UP_1;
}
if (step == WIDTH * WIDTH - 1 - count_barrier) //如果走到了最后一步,还只剩下最后一个格子的情况
{
next_step[3].num = 1; //将值置为一个正数(不一定要是1),以便于最后一步可执行 。否则最后一步会检测到无路可走
}
}
if (x + 1 <= WIDTH && y + 2 <= WIDTH && matrix[x + 1][y + 2].isvisited == false && matrix[x][y + 1].value == 0) // 右2下1可以走吗? 并且检测是否有别马脚
{
k++;
if (count < 1)
{
next_step[4].num = get_next(x + 1, y + 2, next_step, step, count + 1);
next_step[4].togo = RIGHT_2_DOWN_1;
if (step == WIDTH * WIDTH - 1 - count_barrier) //如果走到了最后一步,还只剩下最后一个格子的情况
{
next_step[4].num = 1; //将值置为一个正数(不一定要是1),以便于最后一步可执行 。否则最后一步会检测到无路可走
}
}
}
if (x + 2 <= WIDTH && y + 1 <= WIDTH && matrix[x + 2][y + 1].isvisited == false && matrix[x + 1][y].value == 0) // 右1下2可走吗? 并且检测是否有别马脚
{
k++;
if (count < 1)
{
next_step[5].num = get_next(x + 2, y + 1, next_step, step, count + 1);
next_step[5].togo = RIGHT_1_DOWN_2;
if (step == WIDTH * WIDTH - 1 - count_barrier) //如果走到了最后一步,还只剩下最后一个格子的情况
{
next_step[5].num = 1; //将值置为一个正数(不一定要是1),以便于最后一步可执行 。否则最后一步会检测到无路可走
}
}
}
if (x + 2 <= WIDTH && y - 1 > 0 && matrix[x + 2][y - 1].isvisited == false && matrix[x + 1][y].value == 0) // 左1下2可以走吗
{
k++;
if (count < 1)
{
next_step[6].num = get_next(x + 2, y - 1, next_step, step, count + 1);
next_step[6].togo = LEFT_1_DOWN_2;
if (step == WIDTH * WIDTH - 1 - count_barrier) //如果走到了最后一步,还只剩下最后一个格子的情况
{
next_step[6].num = 1; //将值置为一个正数(不一定要是1),以便于最后一步可执行 。否则最后一步会检测到无路可走
}
}
}
if (x + 1 <= WIDTH && y - 2 > 0 && matrix[x + 1][y - 2].isvisited == false && matrix[x][y - 1].value == 0) // 左2下1可以走吗 并且检测是否有别马脚
{
k++;
if (count < 1)
{
next_step[7].num = get_next(x + 1, y - 2, next_step, step, count + 1);
next_step[7].togo = LEFT_2_DOWN_1;
if (step == WIDTH * WIDTH - 1 - count_barrier) //如果走到了最后一步,还只剩下最后一个格子的情况
{
next_step[7].num = 1; //将值置为一个正数(不一定要是1),以便于最后一步可执行 。否则最后一步会检测到无路可走
}
}
}
return k;
}
int Comp(const void *p1, const void *p2) //用于比较结构体数组的快排
{
ARRAY *c = (ARRAY*)p1;
ARRAY *d = (ARRAY*)p2;
return (c->num - d->num);
}
void DFS_optimized_with_greed(int horse_x, int horse_y, int step) //贪心算法优化后的深度遍历搜索
{
printf("我现在在 (%d,%d) 第%d步\n", horse_x, horse_y, step);
path[step - 1].x = horse_x; //将走过的点存入路径数组中,因为step从1开始计数,所以路径数组要从step-1开始计数
path[step - 1].y = horse_y;
matrix[horse_x][horse_y].isvisited = true;
if (step == WIDTH * WIDTH - count_barrier) //如果马将棋盘走满了
{
found++;
printf("找到了一个诶!");
show_path();
}
ARRAY next_step[8]; //用于记录8个方向的位置
if (get_next(horse_x, horse_y, next_step, step, 0) == 0)
{
}
else
{
qsort(next_step, 8, sizeof(next_step[0]), Comp); //对结构体数组进行排序
for (int i = 0; i < 8; i++)
{
if (next_step[i].num != 0)
{
switch (next_step[i].togo)
{
case LEFT_2_UP_1: DFS_optimized_with_greed(horse_x - 1, horse_y - 2, step + 1); break;
case LEFT_1_UP_2: DFS_optimized_with_greed(horse_x - 2, horse_y - 1, step + 1); break;
case RIGHT_1_UP_2:DFS_optimized_with_greed(horse_x - 2, horse_y + 1, step + 1); break;
case RIGHT_2_UP_1:DFS_optimized_with_greed(horse_x - 1, horse_y + 2, step + 1); break;
case RIGHT_2_DOWN_1:DFS_optimized_with_greed(horse_x + 1, horse_y + 2, step + 1); break;
case RIGHT_1_DOWN_2:DFS_optimized_with_greed(horse_x + 2, horse_y + 1, step + 1); break;
case LEFT_1_DOWN_2: DFS_optimized_with_greed(horse_x + 2, horse_y - 1, step + 1); break;
case LEFT_2_DOWN_1: DFS_optimized_with_greed(horse_x + 1, horse_y - 2, step + 1); break;
default:break;
}
}
}
}
matrix[horse_x][horse_y].isvisited = false;
}
int main()
{
int horse_x, horse_y; //horse's x and y
int count = 0; //to count the dog's total steps
bool set_bool = false;
srand((int)time(0));
init();
set_horse_position(horse_x, horse_y, WIDTH / 2, WIDTH / 2);
//set_barrier(3, 4);
//set_barrier(6, 4);
DFS_optimized_with_greed(horse_x, horse_y, 1);
system("pause");
return 0;
}