鱼C论坛

 找回密码
 立即注册
查看: 1237|回复: 9

[已解决]回溯法解决数独问题

[复制链接]
发表于 2018-8-24 08:19:00 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
最近快被这个问题急死了,调试的时候总是卡在回溯这一步,却也不清楚出了什么问题,求教!
  1. #include<iostream>
  2. #include<math.h>
  3. #include<string>
  4. #include<stdio.h>
  5. #include<algorithm>
  6. #include<vector>
  7. #include<time.h>
  8. /*
  9.                 数独问题
  10. */
  11. #define maxn 400000
  12. using namespace std;
  13. int num[9][9];
  14. bool judge(int i, int j, int number)
  15. {
  16.         int i0 = i / 3 * 3;
  17.         int j0 = j / 3 * 3;
  18.         int ii = i; int jj = j;
  19.         //对块判断
  20.         for (i = i0; i < i0 + 3; i++)
  21.         {
  22.                 for (j = j0; j < j0 + 3; j++)
  23.                 {
  24.                         if (num[i][j] == number)return false;
  25.                 }
  26.         }
  27.         j = jj;
  28.         //对列判断
  29.         for (i = 0; i < 9; i++)
  30.         {
  31.                 if (num[i][j] == number)return false;
  32.         }
  33.         i = ii;
  34.         //对行判断
  35.         for (j = 0; j < 9; j++)
  36.         {
  37.                 if (num[i][j] == number)return false;
  38.         }
  39.         return true;
  40. }
  41. bool isfull()
  42. {
  43.         for (int i = 0; i < 9; i++)
  44.         {
  45.                 for (int j = 0; j < 9; j++)
  46.                 {
  47.                         if (num[i][j] == 0)return false;
  48.                 }
  49.         }
  50.         return true;
  51. }
  52. void out()
  53. {
  54.         cout << "结果" << endl;
  55.         for (int i = 0; i < 9; i++)
  56.         {
  57.                 for (int j = 0; j < 9; j++)
  58.                 {
  59.                         cout << num[i][j] << ' ';
  60.                 }
  61.                 cout << endl;
  62.         }
  63. }
  64. void dfs(int i, int j)
  65. {
  66.         /*
  67.         到达底部时,退出,此时i=j=8;
  68.         */
  69.         if (i>= 8&&isfull())
  70.         {       
  71.                 out();
  72.                 return;
  73.         }
  74.         else {
  75.                 int p;
  76.                 for (p = 1; p <= 9; p++)
  77.                 {
  78.                         /*
  79.                         只有行、列、以及块都可以放置这个数的时候我们才放置,
  80.                         不然就退出此次循环,也就是剪掉这个枝
  81.                         */
  82.                         if (num[i][j])
  83.                         {
  84.                                 if (j <8)dfs(i, j + 1);
  85.                                 if (j >= 8)dfs(i + 1, 0);
  86.                                 continue;
  87.                         }
  88.                         if (judge(i,j,p))
  89.                         {
  90.                                 num[i][j] = p;
  91.                                 if (j < 8)dfs(i, j + 1);
  92.                                 if (j >= 8)dfs(i + 1, 0);
  93.                         /*
  94.                         放置后记得还原
  95.                         */
  96.                                 num[i][j] = 0;
  97.                         }
  98.                 }
  99.         }
  100. }
  101. int main()
  102. {
  103.         //输入
  104.         for (int i = 0; i < 9; i++)
  105.         {
  106.                 for (int j = 0; j < 9; j++)
  107.                 {
  108.                         scanf("%d", &num[i][j]);
  109.                 }
  110.         }
  111.         dfs(0, 0);
  112.        
  113.         return 0;
  114. }
复制代码
最佳答案
2018-8-30 18:17:08
本帖最后由 claws0n 于 2018-8-30 18:22 编辑
  1. #include<iostream>

  2. using namespace std;

  3. int puzzle[9][9];

  4. void out()
  5. {
  6.     cout << "Puzzle Solved Successfully!" << endl;
  7.     for (int i = 0; i < 9; i++)
  8.     {
  9.         for (int j = 0; j < 9; j++)
  10.         {
  11.             cout << puzzle[i][j] << ' ';
  12.             if (j == 2 || j == 5)
  13.                cout << "| ";
  14.         }
  15.         cout << endl;
  16.         if (i == 2 || i == 5)
  17.            cout << "**********************\n";
  18.     }
  19. }

  20. bool isfull()
  21. {
  22.     for (int i = 0; i < 9; i++)
  23.     {
  24.         for (int j = 0; j < 9; j++)
  25.         {
  26.             if (puzzle[i][j] == 0)
  27.                 return false;
  28.         }
  29.     }
  30.     return true;
  31. }

  32. bool judge(int i0, int j0, int number)
  33. {
  34.     int x = i0 / 3 * 3;
  35.     int y = j0 / 3 * 3;
  36.     int i, j;
  37.    
  38.     for (i = x; i < x+3 && i < 9; i++)
  39.     {
  40.         for (j = y; j < y+3 && j < 9; j++)
  41.         {
  42.             if (puzzle[i][j] == number)
  43.                 return false;
  44.         }
  45.     }

  46.     for (x = 0; x < 9; x++)
  47.     {
  48.         if (puzzle[x][j0] == number)
  49.             return false;
  50.     }

  51.     for (y = 0; y < 9; y++)
  52.     {
  53.         if (puzzle[i0][y] == number)
  54.             return false;
  55.     }
  56.     return true;
  57. }

  58. void solveSudoku(int i, int j)
  59. {
  60.     if ( isfull() )
  61.     {      
  62.         out();
  63.         return;
  64.     }
  65.     else
  66.         {
  67.         if (puzzle[i][j])
  68.         {
  69.             if (j <8) {j++;}
  70.             else {i++; j = 0;}
  71.             solveSudoku(i,j);
  72.         }
  73.         else
  74.         {
  75.                 int p;
  76.                 for (p = 1; p <= 9; p++)
  77.                 {
  78.                     if (judge(i,j,p))
  79.                     {
  80.                         puzzle[i][j] = p;
  81.                         solveSudoku(i,j);
  82.                     }
  83.                 }
  84.                 puzzle[i][j] = 0;
  85.             }
  86.     }
  87. }

  88. int main()
  89. {

  90.         puzzle[0][0] = 0;
  91.         puzzle[0][1] = 0;
  92.         puzzle[0][2] = 5;
  93.         puzzle[0][3] = 3;
  94.         puzzle[0][4] = 0;
  95.         puzzle[0][5] = 0;
  96.         puzzle[0][6] = 0;
  97.         puzzle[0][7] = 0;
  98.         puzzle[0][8] = 0;
  99.        
  100.         puzzle[1][0] = 8;
  101.         puzzle[1][1] = 0;
  102.         puzzle[1][2] = 0;
  103.         puzzle[1][3] = 0;
  104.         puzzle[1][4] = 0;
  105.         puzzle[1][5] = 0;
  106.         puzzle[1][6] = 0;
  107.         puzzle[1][7] = 2;
  108.         puzzle[1][8] = 0;
  109.        
  110.         puzzle[2][0] = 0;
  111.         puzzle[2][1] = 7;
  112.         puzzle[2][2] = 0;
  113.         puzzle[2][3] = 0;
  114.         puzzle[2][4] = 1;
  115.         puzzle[2][5] = 0;
  116.         puzzle[2][6] = 5;
  117.         puzzle[2][7] = 0;
  118.         puzzle[2][8] = 0;
  119.        
  120.         puzzle[3][0] = 4;
  121.         puzzle[3][1] = 0;
  122.         puzzle[3][2] = 0;
  123.         puzzle[3][3] = 0;
  124.         puzzle[3][4] = 0;
  125.         puzzle[3][5] = 5;
  126.         puzzle[3][6] = 3;
  127.         puzzle[3][7] = 0;
  128.         puzzle[3][8] = 0;
  129.        
  130.         puzzle[4][0] = 0;
  131.         puzzle[4][1] = 1;
  132.         puzzle[4][2] = 0;
  133.         puzzle[4][3] = 0;
  134.         puzzle[4][4] = 7;
  135.         puzzle[4][5] = 0;
  136.         puzzle[4][6] = 0;
  137.         puzzle[4][7] = 0;
  138.         puzzle[4][8] = 6;
  139.        
  140.         puzzle[5][0] = 0;
  141.         puzzle[5][1] = 0;
  142.         puzzle[5][2] = 3;
  143.         puzzle[5][3] = 2;
  144.         puzzle[5][4] = 0;
  145.         puzzle[5][5] = 0;
  146.         puzzle[5][6] = 0;
  147.         puzzle[5][7] = 8;
  148.         puzzle[5][8] = 0;
  149.        
  150.         puzzle[6][0] = 0;
  151.         puzzle[6][1] = 6;
  152.         puzzle[6][2] = 0;
  153.         puzzle[6][3] = 5;
  154.         puzzle[6][4] = 0;
  155.         puzzle[6][5] = 0;
  156.         puzzle[6][6] = 0;
  157.         puzzle[6][7] = 0;
  158.         puzzle[6][8] = 9;
  159.        
  160.         puzzle[7][0] = 0;
  161.         puzzle[7][1] = 0;
  162.         puzzle[7][2] = 4;
  163.         puzzle[7][3] = 0;
  164.         puzzle[7][4] = 0;
  165.         puzzle[7][5] = 0;
  166.         puzzle[7][6] = 0;
  167.         puzzle[7][7] = 3;
  168.         puzzle[7][8] = 0;
  169.        
  170.         puzzle[8][0] = 0;
  171.         puzzle[8][1] = 0;
  172.         puzzle[8][2] = 0;
  173.         puzzle[8][3] = 0;
  174.         puzzle[8][4] = 0;
  175.         puzzle[8][5] = 9;
  176.         puzzle[8][6] = 7;
  177.         puzzle[8][7] = 0;
  178.         puzzle[8][8] = 0;
  179.        
  180.     printf("Solving sudoku puzzle.....\n");
  181.     solveSudoku(0,0);

  182.     return 0;
  183. }
复制代码

想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

 楼主| 发表于 2018-8-24 08:20:21 | 显示全部楼层
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

 楼主| 发表于 2018-8-24 08:21:56 | 显示全部楼层
附上测试用例一个:
0 0 5 3 0 0 0 0 0
8 0 0 0 0 0 0 2 0
0 7 0 0 1 0 5 0 0
4 0 0 0 0 5 3 0 0
0 1 0 0 7 0 0 0 6
0 0 3 2 0 0 0 8 0
0 6 0 5 0 0 0 0 9
0 0 4 0 0 0 0 3 0
0 0 0 0 0 9 7 0 0
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2018-8-24 19:57:04 | 显示全部楼层
@各路大神
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2018-8-30 15:16:23 | 显示全部楼层

不是不想回,算法没有多少人会
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2018-8-30 18:17:08 | 显示全部楼层    本楼为最佳答案   
本帖最后由 claws0n 于 2018-8-30 18:22 编辑
  1. #include<iostream>

  2. using namespace std;

  3. int puzzle[9][9];

  4. void out()
  5. {
  6.     cout << "Puzzle Solved Successfully!" << endl;
  7.     for (int i = 0; i < 9; i++)
  8.     {
  9.         for (int j = 0; j < 9; j++)
  10.         {
  11.             cout << puzzle[i][j] << ' ';
  12.             if (j == 2 || j == 5)
  13.                cout << "| ";
  14.         }
  15.         cout << endl;
  16.         if (i == 2 || i == 5)
  17.            cout << "**********************\n";
  18.     }
  19. }

  20. bool isfull()
  21. {
  22.     for (int i = 0; i < 9; i++)
  23.     {
  24.         for (int j = 0; j < 9; j++)
  25.         {
  26.             if (puzzle[i][j] == 0)
  27.                 return false;
  28.         }
  29.     }
  30.     return true;
  31. }

  32. bool judge(int i0, int j0, int number)
  33. {
  34.     int x = i0 / 3 * 3;
  35.     int y = j0 / 3 * 3;
  36.     int i, j;
  37.    
  38.     for (i = x; i < x+3 && i < 9; i++)
  39.     {
  40.         for (j = y; j < y+3 && j < 9; j++)
  41.         {
  42.             if (puzzle[i][j] == number)
  43.                 return false;
  44.         }
  45.     }

  46.     for (x = 0; x < 9; x++)
  47.     {
  48.         if (puzzle[x][j0] == number)
  49.             return false;
  50.     }

  51.     for (y = 0; y < 9; y++)
  52.     {
  53.         if (puzzle[i0][y] == number)
  54.             return false;
  55.     }
  56.     return true;
  57. }

  58. void solveSudoku(int i, int j)
  59. {
  60.     if ( isfull() )
  61.     {      
  62.         out();
  63.         return;
  64.     }
  65.     else
  66.         {
  67.         if (puzzle[i][j])
  68.         {
  69.             if (j <8) {j++;}
  70.             else {i++; j = 0;}
  71.             solveSudoku(i,j);
  72.         }
  73.         else
  74.         {
  75.                 int p;
  76.                 for (p = 1; p <= 9; p++)
  77.                 {
  78.                     if (judge(i,j,p))
  79.                     {
  80.                         puzzle[i][j] = p;
  81.                         solveSudoku(i,j);
  82.                     }
  83.                 }
  84.                 puzzle[i][j] = 0;
  85.             }
  86.     }
  87. }

  88. int main()
  89. {

  90.         puzzle[0][0] = 0;
  91.         puzzle[0][1] = 0;
  92.         puzzle[0][2] = 5;
  93.         puzzle[0][3] = 3;
  94.         puzzle[0][4] = 0;
  95.         puzzle[0][5] = 0;
  96.         puzzle[0][6] = 0;
  97.         puzzle[0][7] = 0;
  98.         puzzle[0][8] = 0;
  99.        
  100.         puzzle[1][0] = 8;
  101.         puzzle[1][1] = 0;
  102.         puzzle[1][2] = 0;
  103.         puzzle[1][3] = 0;
  104.         puzzle[1][4] = 0;
  105.         puzzle[1][5] = 0;
  106.         puzzle[1][6] = 0;
  107.         puzzle[1][7] = 2;
  108.         puzzle[1][8] = 0;
  109.        
  110.         puzzle[2][0] = 0;
  111.         puzzle[2][1] = 7;
  112.         puzzle[2][2] = 0;
  113.         puzzle[2][3] = 0;
  114.         puzzle[2][4] = 1;
  115.         puzzle[2][5] = 0;
  116.         puzzle[2][6] = 5;
  117.         puzzle[2][7] = 0;
  118.         puzzle[2][8] = 0;
  119.        
  120.         puzzle[3][0] = 4;
  121.         puzzle[3][1] = 0;
  122.         puzzle[3][2] = 0;
  123.         puzzle[3][3] = 0;
  124.         puzzle[3][4] = 0;
  125.         puzzle[3][5] = 5;
  126.         puzzle[3][6] = 3;
  127.         puzzle[3][7] = 0;
  128.         puzzle[3][8] = 0;
  129.        
  130.         puzzle[4][0] = 0;
  131.         puzzle[4][1] = 1;
  132.         puzzle[4][2] = 0;
  133.         puzzle[4][3] = 0;
  134.         puzzle[4][4] = 7;
  135.         puzzle[4][5] = 0;
  136.         puzzle[4][6] = 0;
  137.         puzzle[4][7] = 0;
  138.         puzzle[4][8] = 6;
  139.        
  140.         puzzle[5][0] = 0;
  141.         puzzle[5][1] = 0;
  142.         puzzle[5][2] = 3;
  143.         puzzle[5][3] = 2;
  144.         puzzle[5][4] = 0;
  145.         puzzle[5][5] = 0;
  146.         puzzle[5][6] = 0;
  147.         puzzle[5][7] = 8;
  148.         puzzle[5][8] = 0;
  149.        
  150.         puzzle[6][0] = 0;
  151.         puzzle[6][1] = 6;
  152.         puzzle[6][2] = 0;
  153.         puzzle[6][3] = 5;
  154.         puzzle[6][4] = 0;
  155.         puzzle[6][5] = 0;
  156.         puzzle[6][6] = 0;
  157.         puzzle[6][7] = 0;
  158.         puzzle[6][8] = 9;
  159.        
  160.         puzzle[7][0] = 0;
  161.         puzzle[7][1] = 0;
  162.         puzzle[7][2] = 4;
  163.         puzzle[7][3] = 0;
  164.         puzzle[7][4] = 0;
  165.         puzzle[7][5] = 0;
  166.         puzzle[7][6] = 0;
  167.         puzzle[7][7] = 3;
  168.         puzzle[7][8] = 0;
  169.        
  170.         puzzle[8][0] = 0;
  171.         puzzle[8][1] = 0;
  172.         puzzle[8][2] = 0;
  173.         puzzle[8][3] = 0;
  174.         puzzle[8][4] = 0;
  175.         puzzle[8][5] = 9;
  176.         puzzle[8][6] = 7;
  177.         puzzle[8][7] = 0;
  178.         puzzle[8][8] = 0;
  179.        
  180.     printf("Solving sudoku puzzle.....\n");
  181.     solveSudoku(0,0);

  182.     return 0;
  183. }
复制代码

想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2018-9-1 22:14:12 | 显示全部楼层
77 - 79 可以换成这样,比较好
  1. if (j < 8) {solveSudoku(i, j+1);}
  2. else {solveSudoku(i+1, 0);}
复制代码
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

头像被屏蔽
发表于 2018-9-21 13:30:45 | 显示全部楼层
提示: 作者被禁止或删除 内容自动屏蔽
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2018-9-21 16:17:29 | 显示全部楼层

回帖奖励 +5 鱼币

可以的
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2018-9-22 11:15:53 | 显示全部楼层

回帖奖励 +5 鱼币

学习
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2024-4-25 19:33

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表