鱼C论坛

 找回密码
 立即注册
查看: 962|回复: 2

[已解决]井字棋

[复制链接]
发表于 2023-11-9 15:18:06 | 显示全部楼层 |阅读模式

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

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

x
我的程序测试的时候和电脑下的时候一直都无法出现平局的情况(电脑总是下着下着就让我赢了),请问哪位大佬可以用AI帮我测试一下,感谢大佬!
  1. #include <stdio.h>
  2. #include <string.h>
  3. #include <stdbool.h>
  4. #include <stdlib.h>
  5. #include <time.h>

  6. void printArray(char arr[3][3])
  7. {
  8.     int i,j;
  9.     for (i=0; i<3; i++)
  10.     {
  11.         for (j=0; j<3; j++)
  12.         {
  13.             printf("%c ", arr[i][j]);
  14.         }
  15.         printf("\n");
  16.     }
  17. }

  18. bool humanwin(char arr[3][3])
  19. {
  20.     for (int i = 0; i < 3; i++)
  21.     {
  22.         if (arr[ i][0] == 'X' && arr[ i][1] == 'X' && arr[ i][2] == 'X')
  23.         {
  24.             return true;
  25.         }
  26.     }
  27.     for (int j = 0; j < 3; j++)
  28.     {
  29.         if (arr[0][j] == 'X' && arr[1][j] == 'X' && arr[2][j] == 'X')
  30.         {
  31.             return true;
  32.         }
  33.         
  34.     }
  35.     if (arr[0][0] == 'X' && arr[1][1] == 'X' && arr[2][2] == 'X')
  36.     {
  37.         return true;
  38.     }
  39.     if (arr[0][2] == 'X' && arr[1][1] == 'X' && arr[2][0] == 'X')
  40.     {
  41.         return true;
  42.     }
  43.     return false;
  44. }

  45. bool computerwin(char arr[3][3])
  46. {
  47.     for (int i = 0; i < 3; i++)
  48.     {
  49.         if (arr[ i][0] == 'O' && arr[ i][1] == 'O' && arr[ i][2] == 'O')
  50.         {
  51.             return true;
  52.         }
  53.     }
  54.     for (int j = 0; j < 3; j++)
  55.     {
  56.         if (arr[0][j] == 'O' && arr[1][j] == 'O' && arr[2][j] == 'O')
  57.         {
  58.             return true;
  59.         }
  60.         
  61.     }
  62.     if (arr[0][0] == 'O' && arr[1][1] == 'O' && arr[2][2] == 'O')
  63.     {
  64.         return true;
  65.     }
  66.     if (arr[0][2] == 'O' && arr[1][1] == 'O' && arr[2][0] == 'O')
  67.     {
  68.         return true;
  69.     }
  70.     return false;
  71. }

  72. bool nowin(char arr[3][3])
  73. {
  74.     if (arr[0][0] != '.' && arr[0][1] != '.' && arr[0][2] != '.' && arr[1][0] != '.' && arr[1][1] != '.' && arr[1][2] != '.' && arr[2][0] != '.' && arr[2][1] != '.' && arr[2][2] != '.')
  75.     {
  76.         return true;
  77.     }
  78.     else
  79.     {
  80.         return false;
  81.     }
  82. }
  83.    

  84. void calculateRowAndCol(int x, int* row, int* col)
  85. {
  86.     *row = 2 - ((x-1) / 3);
  87.     *col = (x-1) % 3;
  88. }


  89. int main()
  90. {
  91.     char board_now[3][3] = {{'.','.','.'},{'.','.','.'},{'.','.','.'}};
  92.     int i,x,y,row,col;
  93.     for (i=0;i<9;i++)
  94.     {
  95.         for (i=0;i<5;i++)
  96.         {
  97.             printArray(board_now);
  98.             printf("human input:");
  99.             scanf("%i",&x);
  100.             calculateRowAndCol(x, &row, &col);
  101.             if (board_now[row][col] == '.')
  102.             {
  103.                 board_now[row][col] = 'X';
  104.                 printf("\033[2J\033[H");
  105.                 break;
  106.             }
  107.             else
  108.             {
  109.                 printf("\033[2J\033[H");
  110.                 printf("you have done this.please input again.\n");
  111.                 i = i-1;
  112.             }   
  113.         }
  114.         if (humanwin(board_now))
  115.         {
  116.             printArray(board_now);
  117.             printf("human win.\n");
  118.             break;
  119.         }
  120.         for (i=0;i<4;i++)
  121.         {
  122.             srand(time(NULL));
  123.             y = rand()%(9+1);
  124.             calculateRowAndCol(y, &row, &col);
  125.             if (board_now[row][col] == '.')
  126.             {
  127.                 board_now[row][col] = 'O';
  128.                 printf("\033[2J\033[H");
  129.                 break;
  130.             }
  131.             else
  132.             {
  133.                 i = i-1;
  134.                 if (nowin(board_now))
  135.                 {
  136.                     printf("nobady win.");
  137.                     break;
  138.                 }
  139.             }
  140.         }
  141.         if (computerwin(board_now))
  142.         {
  143.             printArray(board_now);
  144.             printf("computer win.\n");
  145.             break;
  146.         }  
  147.     }
  148.     return 0;     
  149. }
复制代码
最佳答案
2023-11-9 15:21:00
这个程序的问题在于计算机的移动是随机的,没有使用任何AI算法来进行决策。为了实现一个可以与玩家下井字棋并且可以出现平局的AI,你可以使用一个简单的算法,例如最小最大算法(Minimax algorithm)。

最小最大算法是一种用于决策的算法,它可以找到最佳的移动来最大化自己的得分或最小化对手的得分。在井字棋中,你可以将计算机的角色定义为最小化得分的一方,玩家的角色定义为最大化得分的一方。

下面是一个使用最小最大算法的示例代码:

  1. #include <stdio.h>
  2. #include <stdbool.h>
  3. #include <stdlib.h>
  4. #include <time.h>

  5. #define SIZE 3

  6. void printArray(char arr[SIZE][SIZE])
  7. {
  8.     for (int i=0; i<SIZE; i++)
  9.     {
  10.         for (int j=0; j<SIZE; j++)
  11.         {
  12.             printf("%c ", arr[ i][j]);
  13.         }
  14.         printf("\n");
  15.     }
  16. }

  17. bool humanwin(char arr[SIZE][SIZE])
  18. {
  19.     for (int i = 0; i < SIZE; i++)
  20.     {
  21.         if (arr[ i][0] == 'X' && arr[ i][1] == 'X' && arr[ i][2] == 'X')
  22.         {
  23.             return true;
  24.         }
  25.     }
  26.     for (int j = 0; j < SIZE; j++)
  27.     {
  28.         if (arr[0][j] == 'X' && arr[1][j] == 'X' && arr[2][j] == 'X')
  29.         {
  30.             return true;
  31.         }
  32.     }
  33.     if (arr[0][0] == 'X' && arr[1][1] == 'X' && arr[2][2] == 'X')
  34.     {
  35.         return true;
  36.     }
  37.     if (arr[0][2] == 'X' && arr[1][1] == 'X' && arr[2][0] == 'X')
  38.     {
  39.         return true;
  40.     }
  41.     return false;
  42. }

  43. bool computerwin(char arr[SIZE][SIZE])
  44. {
  45.     for (int i = 0; i < SIZE; i++)
  46.     {
  47.         if (arr[ i][0] == 'O' && arr[ i][1] == 'O' && arr[ i][2] == 'O')
  48.         {
  49.             return true;
  50.         }
  51.     }
  52.     for (int j = 0; j < SIZE; j++)
  53.     {
  54.         if (arr[0][j] == 'O' && arr[1][j] == 'O' && arr[2][j] == 'O')
  55.         {
  56.             return true;
  57.         }
  58.     }
  59.     if (arr[0][0] == 'O' && arr[1][1] == 'O' && arr[2][2] == 'O')
  60.     {
  61.         return true;
  62.     }
  63.     if (arr[0][2] == 'O' && arr[1][1] == 'O' && arr[2][0] == 'O')
  64.     {
  65.         return true;
  66.     }
  67.     return false;
  68. }

  69. bool nowin(char arr[SIZE][SIZE])
  70. {
  71.     for (int i = 0; i < SIZE; i++)
  72.     {
  73.         for (int j = 0; j < SIZE; j++)
  74.         {
  75.             if (arr[ i][j] == '.')
  76.             {
  77.                 return false;
  78.             }
  79.         }
  80.     }
  81.     return true;
  82. }

  83. int evaluate(char arr[SIZE][SIZE])
  84. {
  85.     if (humanwin(arr))
  86.     {
  87.         return -1;
  88.     }
  89.     else if (computerwin(arr))
  90.     {
  91.         return 1;
  92.     }
  93.     else
  94.     {
  95.         return 0;
  96.     }
  97. }

  98. int minimax(char arr[SIZE][SIZE], int depth, bool isMaximizer)
  99. {
  100.     int score = evaluate(arr);

  101.     if (score == 1)
  102.     {
  103.         return score;
  104.     }
  105.     if (score == -1)
  106.     {
  107.         return score;
  108.     }
  109.     if (nowin(arr))
  110.     {
  111.         return 0;
  112.     }

  113.     if (isMaximizer)
  114.     {
  115.         int best = -1000;

  116.         for (int i = 0; i < SIZE; i++)
  117.         {
  118.             for (int j = 0; j < SIZE; j++)
  119.             {
  120.                 if (arr[ i][j] == '.')
  121.                 {
  122.                     arr[ i][j] = 'O';
  123.                     best = max(best, minimax(arr, depth + 1, !isMaximizer));
  124.                     arr[ i][j] = '.';
  125.                 }
  126.             }
  127.         }

  128.         return best;
  129.     }
  130.     else
  131.     {
  132.         int best = 1000;

  133.         for (int i = 0; i < SIZE; i++)
  134.         {
  135.             for (int j = 0; j < SIZE; j++)
  136.             {
  137.                 if (arr[ i][j] == '.')
  138.                 {
  139.                     arr[ i][j] = 'X';
  140.                     best = min(best, minimax(arr, depth + 1, !isMaximizer));
  141.                     arr[ i][j] = '.';
  142.                 }
  143.             }
  144.         }

  145.         return best;
  146.     }
  147. }

  148. void findBestMove(char arr[SIZE][SIZE])
  149. {
  150.     int bestVal = -1000;
  151.     int bestRow = -1;
  152.     int bestCol = -1;

  153.     for (int i = 0; i < SIZE; i++)
  154.     {
  155.         for (int j = 0; j < SIZE; j++)
  156.         {
  157.             if (arr[ i][j] == '.')
  158.             {
  159.                 arr[ i][j] = 'O';
  160.                 int moveVal = minimax(arr, 0, false);
  161.                 arr[ i][j] = '.';

  162.                 if (moveVal > bestVal)
  163.                 {
  164.                     bestRow = i;
  165.                     bestCol = j;
  166.                     bestVal = moveVal;
  167.                 }
  168.             }
  169.         }
  170.     }

  171.     arr[bestRow][bestCol] = 'O';
  172. }

  173. int main()
  174. {
  175.     char board_now[SIZE][SIZE] = {{'.','.','.'},{'.','.','.'},{'.','.','.'}};
  176.     int x, row, col;

  177.     while (!nowin(board_now))
  178.     {
  179.         printArray(board_now);
  180.         printf("human input:");
  181.         scanf("%i", &x);
  182.         calculateRowAndCol(x, &row, &col);

  183.         if (board_now[row][col] == '.')
  184.         {
  185.             board_now[row][col] = 'X';
  186.             printf("\033[2J\033[H");
  187.         }
  188.         else
  189.         {
  190.             printf("\033[2J\033[H");
  191.             printf("you have done this.please input again.\n");
  192.             continue;
  193.         }

  194.         if (humanwin(board_now))
  195.         {
  196.             printArray(board_now);
  197.             printf("human win.\n");
  198.             break;
  199.         }

  200.         findBestMove(board_now);

  201.         if (computerwin(board_now))
  202.         {
  203.             printArray(board_now);
  204.             printf("computer win.\n");
  205.             break;
  206.         }
  207.     }

  208.     if (nowin(board_now))
  209.     {
  210.         printArray(board_now);
  211.         printf("nobady win.\n");
  212.     }

  213.     return 0;
  214. }
复制代码


这个代码使用了递归的最小最大算法来决策计算机的下一步棋。它会遍历所有可能的下一步棋,并计算出每个可能的得分,然后选择得分最高的一步作为计算机的下一步棋。

希望这个代码可以帮助你解决问题!
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

发表于 2023-11-9 15:21:00 | 显示全部楼层    本楼为最佳答案   
这个程序的问题在于计算机的移动是随机的,没有使用任何AI算法来进行决策。为了实现一个可以与玩家下井字棋并且可以出现平局的AI,你可以使用一个简单的算法,例如最小最大算法(Minimax algorithm)。

最小最大算法是一种用于决策的算法,它可以找到最佳的移动来最大化自己的得分或最小化对手的得分。在井字棋中,你可以将计算机的角色定义为最小化得分的一方,玩家的角色定义为最大化得分的一方。

下面是一个使用最小最大算法的示例代码:

  1. #include <stdio.h>
  2. #include <stdbool.h>
  3. #include <stdlib.h>
  4. #include <time.h>

  5. #define SIZE 3

  6. void printArray(char arr[SIZE][SIZE])
  7. {
  8.     for (int i=0; i<SIZE; i++)
  9.     {
  10.         for (int j=0; j<SIZE; j++)
  11.         {
  12.             printf("%c ", arr[ i][j]);
  13.         }
  14.         printf("\n");
  15.     }
  16. }

  17. bool humanwin(char arr[SIZE][SIZE])
  18. {
  19.     for (int i = 0; i < SIZE; i++)
  20.     {
  21.         if (arr[ i][0] == 'X' && arr[ i][1] == 'X' && arr[ i][2] == 'X')
  22.         {
  23.             return true;
  24.         }
  25.     }
  26.     for (int j = 0; j < SIZE; j++)
  27.     {
  28.         if (arr[0][j] == 'X' && arr[1][j] == 'X' && arr[2][j] == 'X')
  29.         {
  30.             return true;
  31.         }
  32.     }
  33.     if (arr[0][0] == 'X' && arr[1][1] == 'X' && arr[2][2] == 'X')
  34.     {
  35.         return true;
  36.     }
  37.     if (arr[0][2] == 'X' && arr[1][1] == 'X' && arr[2][0] == 'X')
  38.     {
  39.         return true;
  40.     }
  41.     return false;
  42. }

  43. bool computerwin(char arr[SIZE][SIZE])
  44. {
  45.     for (int i = 0; i < SIZE; i++)
  46.     {
  47.         if (arr[ i][0] == 'O' && arr[ i][1] == 'O' && arr[ i][2] == 'O')
  48.         {
  49.             return true;
  50.         }
  51.     }
  52.     for (int j = 0; j < SIZE; j++)
  53.     {
  54.         if (arr[0][j] == 'O' && arr[1][j] == 'O' && arr[2][j] == 'O')
  55.         {
  56.             return true;
  57.         }
  58.     }
  59.     if (arr[0][0] == 'O' && arr[1][1] == 'O' && arr[2][2] == 'O')
  60.     {
  61.         return true;
  62.     }
  63.     if (arr[0][2] == 'O' && arr[1][1] == 'O' && arr[2][0] == 'O')
  64.     {
  65.         return true;
  66.     }
  67.     return false;
  68. }

  69. bool nowin(char arr[SIZE][SIZE])
  70. {
  71.     for (int i = 0; i < SIZE; i++)
  72.     {
  73.         for (int j = 0; j < SIZE; j++)
  74.         {
  75.             if (arr[ i][j] == '.')
  76.             {
  77.                 return false;
  78.             }
  79.         }
  80.     }
  81.     return true;
  82. }

  83. int evaluate(char arr[SIZE][SIZE])
  84. {
  85.     if (humanwin(arr))
  86.     {
  87.         return -1;
  88.     }
  89.     else if (computerwin(arr))
  90.     {
  91.         return 1;
  92.     }
  93.     else
  94.     {
  95.         return 0;
  96.     }
  97. }

  98. int minimax(char arr[SIZE][SIZE], int depth, bool isMaximizer)
  99. {
  100.     int score = evaluate(arr);

  101.     if (score == 1)
  102.     {
  103.         return score;
  104.     }
  105.     if (score == -1)
  106.     {
  107.         return score;
  108.     }
  109.     if (nowin(arr))
  110.     {
  111.         return 0;
  112.     }

  113.     if (isMaximizer)
  114.     {
  115.         int best = -1000;

  116.         for (int i = 0; i < SIZE; i++)
  117.         {
  118.             for (int j = 0; j < SIZE; j++)
  119.             {
  120.                 if (arr[ i][j] == '.')
  121.                 {
  122.                     arr[ i][j] = 'O';
  123.                     best = max(best, minimax(arr, depth + 1, !isMaximizer));
  124.                     arr[ i][j] = '.';
  125.                 }
  126.             }
  127.         }

  128.         return best;
  129.     }
  130.     else
  131.     {
  132.         int best = 1000;

  133.         for (int i = 0; i < SIZE; i++)
  134.         {
  135.             for (int j = 0; j < SIZE; j++)
  136.             {
  137.                 if (arr[ i][j] == '.')
  138.                 {
  139.                     arr[ i][j] = 'X';
  140.                     best = min(best, minimax(arr, depth + 1, !isMaximizer));
  141.                     arr[ i][j] = '.';
  142.                 }
  143.             }
  144.         }

  145.         return best;
  146.     }
  147. }

  148. void findBestMove(char arr[SIZE][SIZE])
  149. {
  150.     int bestVal = -1000;
  151.     int bestRow = -1;
  152.     int bestCol = -1;

  153.     for (int i = 0; i < SIZE; i++)
  154.     {
  155.         for (int j = 0; j < SIZE; j++)
  156.         {
  157.             if (arr[ i][j] == '.')
  158.             {
  159.                 arr[ i][j] = 'O';
  160.                 int moveVal = minimax(arr, 0, false);
  161.                 arr[ i][j] = '.';

  162.                 if (moveVal > bestVal)
  163.                 {
  164.                     bestRow = i;
  165.                     bestCol = j;
  166.                     bestVal = moveVal;
  167.                 }
  168.             }
  169.         }
  170.     }

  171.     arr[bestRow][bestCol] = 'O';
  172. }

  173. int main()
  174. {
  175.     char board_now[SIZE][SIZE] = {{'.','.','.'},{'.','.','.'},{'.','.','.'}};
  176.     int x, row, col;

  177.     while (!nowin(board_now))
  178.     {
  179.         printArray(board_now);
  180.         printf("human input:");
  181.         scanf("%i", &x);
  182.         calculateRowAndCol(x, &row, &col);

  183.         if (board_now[row][col] == '.')
  184.         {
  185.             board_now[row][col] = 'X';
  186.             printf("\033[2J\033[H");
  187.         }
  188.         else
  189.         {
  190.             printf("\033[2J\033[H");
  191.             printf("you have done this.please input again.\n");
  192.             continue;
  193.         }

  194.         if (humanwin(board_now))
  195.         {
  196.             printArray(board_now);
  197.             printf("human win.\n");
  198.             break;
  199.         }

  200.         findBestMove(board_now);

  201.         if (computerwin(board_now))
  202.         {
  203.             printArray(board_now);
  204.             printf("computer win.\n");
  205.             break;
  206.         }
  207.     }

  208.     if (nowin(board_now))
  209.     {
  210.         printArray(board_now);
  211.         printf("nobady win.\n");
  212.     }

  213.     return 0;
  214. }
复制代码


这个代码使用了递归的最小最大算法来决策计算机的下一步棋。它会遍历所有可能的下一步棋,并计算出每个可能的得分,然后选择得分最高的一步作为计算机的下一步棋。

希望这个代码可以帮助你解决问题!
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2023-11-9 15:29:14 | 显示全部楼层
isdkz 发表于 2023-11-9 15:21
这个程序的问题在于计算机的移动是随机的,没有使用任何AI算法来进行决策。为了实现一个可以与玩家下井字棋 ...

额,并没有解决我的问题,不过算了自己手动运行测试吧
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

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

本版积分规则

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

GMT+8, 2025-4-21 14:51

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

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