数据结构-九宫格(回溯法)

数独,是源自18世纪瑞士的一种数学游戏。是一种运用纸、笔进行演算的逻辑游戏。玩家需要根据9×9盘面上的已知数字,推理出所有剩余空格的数字,并满足每一行、每一列、每一个粗线宫(3*3)内的数字均含1-9,不重复。

数独盘面是个九宫,每一宫又分为九个小格。在这八十一格中给出一定的已知数字和解题条件,利用逻辑和推理,在其他的空格上填入1-9的数字。使1-9每个数字在每一行、每一列和每一宫中都只出现一次,所以又称“九宫格”。
利用算法解数独,主要采用了回溯法。思路如下:

1 .遍历已生成的数独二维数组,得出空白格子的数目。

2 .从第一个空白格子开始,利用数独的规范,对比同一列,同一行,以及同一个九宫格的数字,找出其所有可行解,存入数组(利用整形变量的位运算,会有更高的效率)。利用最后一个可行解,进行下一步运算。

3 .对剩下的格子进行同样的操作。

4 .如遇到无解的情况,则进行回溯操作。继续重复上述运算。

5 .当所有空白格子填满,所得结果,即为数独的解。

C++代码:
#include
#include
using namespace std;
int map[9][9];
bool isPlace(int count){
int row = count / 9;
int col = count % 9;
int j;
//同一行
for(j = 0; j < 9; ++j){
if(map[row][j] == map[row][col] && j != col){
return false;
}
}
//同一列
for(j = 0; j < 9; ++j){
if(map[j][col] == map[row][col] && j != row){
return false;
}
}
//同一小格
int tempRow = row / 3 * 3;
int tempCol = col / 3 * 3;
for(j = tempRow; j < tempRow + 3;++j){
for(int k = tempCol; k < tempCol + 3; ++k){
if(map[j][k] == map[row][col] && j != row && k != col){
return false;
}
}
}
return true;
}
void backtrace(int count){
if(count == 81){
cout<<“结果:”<<endl;
for(int i = 0; i < 9; ++i){
for(int j = 0; j < 9; ++j){
cout<<map[i][j]<<" ";
}
cout<<endl;
}
return;
}
int row = count / 9;
int col = count % 9;
if(map[row][col] == 0){
for(int i = 1; i <= 9; ++i){
map[row][col] = i;//赋值
if(isPlace(count)){//可以放
backtrace(count+1);//进入下一层
}
}
map[row][col] = 0;//回溯
}else{
backtrace(count+1);
}
}
int main()
{
for(int i = 0; i < 9; ++i){
for(int j = 0; j < 9; ++j){
cin>>map[i][j];
}
}
backtrace(0);
return 0;
}

  • 3
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
回溯法在解决九宫格问题(即数独问题)时非常常见。九宫格问题是一个填充数独空白格子的问题,要求每一行、每一列和每一个九宫格内都没有重复的数字。 下面是一个使用回溯法解决九宫格问题的示例代码: ```python def solve_sudoku(board): if not find_empty_cell(board): return True row, col = find_empty_cell(board) for num in range(1, 10): if is_valid(board, row, col, num): board[row][col] = num if solve_sudoku(board): return True board[row][col] = 0 return False def find_empty_cell(board): for i in range(9): for j in range(9): if board[i][j] == 0: return i, j return None def is_valid(board, row, col, num): # 检查行是否合法 for i in range(9): if board[row][i] == num: return False # 检查列是否合法 for i in range(9): if board[i][col] == num: return False # 检查小九宫格是否合法 start_row = (row // 3) * 3 # 小九宫格的起始行索引 start_col = (col // 3) * 3 # 小九宫格的起始列索引 for i in range(3): for j in range(3): if board[start_row + i][start_col + j] == num: return False return True ``` 这段代码使用递归的方式进行回溯。`solve_sudoku` 函数会先找到一个空白格子,然后尝试填入数字并判断是否合法。如果合法,则继续递归地尝试下一个空白格子;如果不合法,则回溯到上一个空白格子,尝试其他数字。如果所有格子都填满且合法,就找到了一个解,返回 True。如果无法找到解,则返回 False。 你可以将九宫格问题的初始状态传递给 `solve_sudoku` 函数,它将会返回解决后的九宫格。示例调用代码如下: ```python board = [ [5, 3, 0, 0, 7, 0, 0, 0, 0], [6, 0, 0, 1, 9, 5, 0, 0, 0], [0, 9, 8, 0, 0, 0, 0, 6, 0], [8, 0, 0, 0, 6, 0, 0, 0, 3], [4, 0, 0, 8, 0, 3, 0, 0, 1], [7, 0, 0, 0, 2, 0, 0, 0, 6], [0, 6, 0, 0, 0, 0, 2, 8, 0], [0, 0, 0, 4, 1, 9, 0, 0, 5], [0, 0, 0, 0, 8, 0, 0, 7, 9] ] solve_sudoku(board) for row in board: print(row) ``` 这段代码会输出解决后的九宫格。空白格子用0表示。 希望这个例子对你有帮助!如果你还有其他问题,请随时问我。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值