最近沉迷于《NetHack》、《DCSS》等字符游戏,对其很感兴趣,于是用C语言写了个字符界面的井字棋小游戏。欢迎大家指教。

编写时遇到了一些问题,我原先准备用循环,直到读取到正确的输入。可该死的getchar函数,在读取后,又把回车又传给下次循环,我不得不对其进行处理。

设定井字棋的AI时,有个有趣的地方就是,先下四个角比先下中心优势更大,这违背了我以前的直觉。

  1 #include <stdio.h>
2 #include <ctype.h>
3 #include <string.h>
4 #include <stdlib.h>
5 #include <time.h>
6
7 void drawBoard(char *board) //绘制棋盘
8 {
9 printf("%c|%c|%c\n", board[7], board[8], board[9]);
10 puts("-+-+-");
11 printf("%c|%c|%c\n", board[4], board[5], board[6]);
12 puts("-+-+-");
13 printf("%c|%c|%c\n", board[1], board[2], board[3]);
14 puts("-+-+-");
15 }
16
17 char inputPlayerLetter() //玩家选择棋子
18 {
19 char letter;
20 puts("你想用X还是O?");
21 do{
22 letter = toupper(getchar());
23 if (letter == '\n')
24 continue;
25 if (letter != 'X' && letter != 'O')
26 puts("你想用X还是O?");
27 }while (letter != 'X' && letter != 'O');
28
29 return letter;
30
31 }
32
33 void makeMove(char *board,char letter,int move) //落子
34 {
35 board[move] = letter;
36 }
37
38 _Bool isWinner(char *bo, char le) //判定是否获胜
39 {
40 return ((bo[7] == le && bo[8] == le && bo[9] == le) ||
41 (bo[4] == le && bo[5] == le && bo[6] == le) ||
42 (bo[1] == le && bo[2] == le && bo[3] == le) ||
43 (bo[7] == le && bo[4] == le && bo[1] == le) ||
44 (bo[8] == le && bo[5] == le && bo[2] == le) ||
45 (bo[9] == le && bo[6] == le && bo[3] == le) ||
46 (bo[7] == le && bo[5] == le && bo[3] == le) ||
47 (bo[9] == le && bo[5] == le && bo[1] == le));
48 }
49
50 const char getBoardCopy(char *board) //复制棋盘,让电脑预判可能出现的情况
51 {
52 char boardCopy[10];
53 for (int i = 1; i < 10; i++)
54 boardCopy[i] = board[i];
55 return *boardCopy;
56 }
57
58 _Bool isSpaceFree(char *board,int move) //判断棋盘上是否为空
59 {
60 return board[move] == ' ';
61 }
62
63 int getPlayerMove(char *board) //读取玩家棋子移动
64 {
65 puts("你下一步走哪里?(1-9)");
66 int move;
67 do {
68 move = getchar() - '0';
69 if (move == '\n' - '0')
70 continue;
71 if (move < 1 || move > 9 || !isSpaceFree(board, move))
72 puts("你下一步走哪里?(1-9)");
73 }while (move < 1 || move > 9 || !isSpaceFree(board, move));
74 return move;
75 }
76
77 int chooseRandomMoveFromList(char *board,char *movelist, int n) //随机读取计算机可移动的位置
78 {
79 int possibleMove[4]; //每轮选择最多只有四个
80 int j = 0;
81 for (int i = 0; i < n; i++)
82 if (isSpaceFree(board, movelist[i] - '0'))
83 possibleMove[j++] = movelist[i] - '0';
84
85 if (j != 0)
86 return possibleMove[rand()%j];
87 else
88 return 0;
89 }
90
91 int getComputerMove(char board[], char computerLetter) //获得计算机的移动
92 {
93 char playerLetter;
94 char boardCopy[10];
95 if (computerLetter == 'X') //根据计算机的棋子,判断玩家棋子
96 playerLetter = 'O';
97 else
98 playerLetter = 'X';
99
100 for (int i = 1; i < 10; i++){ //如果下一步可获胜,下那一步
101 strcpy(boardCopy, board);
102 if (isSpaceFree(boardCopy, i)) {
103 makeMove(boardCopy, computerLetter,i);
104 if (isWinner(boardCopy, computerLetter))
105 return i;
106 }
107 }
108
109 for (int i = 1; i < 10; i++){ //如果下一步玩家会获胜,占那个位置
110 strcpy(boardCopy,board);
111 if (isSpaceFree(boardCopy, i)) {
112 makeMove(boardCopy, playerLetter,i);
113 if (isWinner(boardCopy, playerLetter))
114 return i;
115 }
116 }
117
118 int move; //如果下一步不是决胜步
119 move = chooseRandomMoveFromList(board, "1379", 4); //四个角优先
120 if (move != 0)
121 return move;
122
123 move = chooseRandomMoveFromList(board, "5", 1); //中间
124 if (move != 0)
125 return move;
126
127 return chooseRandomMoveFromList(board, "2468", 4); //剩下的位置
128 }
129
130 _Bool isBoardFull(char *board) //判断棋盘是否满了
131 {
132 for (int i = 1; i < 10; i++)
133 if (isSpaceFree(board, i))
134 return 0;
135 return 1;
136 }
137
138 _Bool isAgain() //再来一局
139 {
140 char again;
141 do{
142 again = tolower(getchar());
143 if (again == '\n')
144 continue;
145 if (again != 'n' && again != 'y')
146 puts("请输入y或n。");
147 }while (again != 'n' && again != 'y');
148
149 if (again == 'y')
150 return 1;
151 else
152 return 0;
153 }
154
155
156 int main()
157 {
158 puts("欢迎来玩井字棋!") ;
159
160 while(1) { //游戏
161 char theBoard[10];
162 for (int i = 1; i < 10; i++) //将棋盘设为空白
163 theBoard[i] = ' ';
164 char playerLetter = inputPlayerLetter(); //获得玩家所选的棋子
165 char computerLetter = (playerLetter == 'X')?'O': 'X' ;
166 //获得计算机的棋子
167 _Bool isTurnPlayer; //设定是否是玩家回合
168 int move;
169 srand((unsigned)time(NULL)); //随机先后手
170 if (rand() % 2) {
171 isTurnPlayer = 0;
172 puts("电脑先走。");
173 }else {
174 isTurnPlayer = 1;
175 puts("玩家先走。");
176 }
177 _Bool gameIsPlaying = 1; //设定游戏是否进行
178
179 while (gameIsPlaying) {
180
181 if (isTurnPlayer) { //如果是玩家回合
182 drawBoard(theBoard);
183 move = getPlayerMove(theBoard);
184 makeMove(theBoard, playerLetter, move);
185
186 if (isWinner(theBoard, playerLetter)) { //如果获胜
187 drawBoard(theBoard);
188 puts("太棒了!你获胜了!");
189 gameIsPlaying = 0;
190 }
191
192 else { //如果平局
193 if (isBoardFull(theBoard)) {
194 drawBoard(theBoard);
195 puts("平局了!");
196 break;
197 }
198 else //设定轮到计算机
199 isTurnPlayer = 0;
200 }
201 }
202
203 else { //轮到计算机了
204 move = getComputerMove(theBoard, computerLetter);
205 makeMove(theBoard, computerLetter, move);
206 if (isWinner(theBoard, computerLetter)) { //如果获胜
207 drawBoard(theBoard);
208 puts("电脑打败了你!你输了。");
209 gameIsPlaying = 0;
210 } else { //平局
211 if (isBoardFull(theBoard)) {
212 drawBoard(theBoard);
213 puts("平局了!");
214 break;
215 } else // 设定轮到玩家
216 isTurnPlayer = 1;
217 }
218 }
219 }
220 puts("再来一局?(yes或no)");
221 if (!isAgain())
222 break;
223 }
224 return 0;
225 }

井字棋小游戏(C语言)的更多相关文章

  1. [LeetCode] Valid Tic-Tac-Toe State 验证井字棋状态

    A Tic-Tac-Toe board is given as a string array board. Return True if and only if it is possible to r ...

  2. JavaScript写的一个带AI的井字棋

    最近有一门课结束了,需要做一个井字棋的游戏,我用JavaScript写了一个.首先界面应该问题不大,用html稍微写一下就可以.主要是人机对弈时的ai算法,如何使电脑方聪明起来,是值得思考一下的.开始 ...

  3. [LeetCode] 794. Valid Tic-Tac-Toe State 验证井字棋状态

    A Tic-Tac-Toe board is given as a string array board. Return True if and only if it is possible to r ...

  4. [HTML5实现人工智能]小游戏《井字棋》发布,据说IQ上200才能赢

    一,什么是TicTacToe(井字棋)   本 游戏 为在下用lufylegend开发的第二款小游戏.此游戏是大家想必大家小时候都玩过,因为玩它很简单,只需要一张草稿纸和一只笔就能开始游戏,所以广受儿 ...

  5. 『HTML5实现人工智能』小游戏《井字棋》发布,据说IQ上200才能赢【算法&代码讲解+资源打包下载】

    一,什么是TicTacToe(井字棋) 本游戏为在下用lufylegend开发的第二款小游戏.此游戏是大家想必大家小时候都玩过,因为玩它很简单,只需要一张草稿纸和一只笔就能开始游戏,所以广受儿童欢迎. ...

  6. Pascal小游戏 井字棋

    一个很经典的井字棋游戏 Pascal源码Chaobs奉上 注意:1.有的FP版本不支持汉语,将会出现乱码.2.别想赢电脑了,平手不错了. 井字过三关: program TicTacToe; uses ...

  7. Java 小游戏 - 井字棋 v1.0 (初步完成) (2018.4.16更新)

      井字棋游戏初步完成 实现功能:输入位置数据->打印棋盘->判断是否胜利->继续游戏/退出游戏 缺点:没有清屏函数   判断胜利方法太过无脑    package MYGAME; ...

  8. 井字棋游戏升级版 - TopTicTacToe项目 简介

    一.游戏简介 井字棋是一款世界闻名的游戏,不用我说,你一定知道它的游戏规则. 这款游戏简单易学,玩起来很有意思,不过已经证明出这款游戏如果两个玩家都足够聪明的话, 是很容易无法分出胜负的,即我们得到的 ...

  9. [CareerCup] 17.2 Tic Tac Toe 井字棋游戏

    17.2 Design an algorithm to figure out if someone has won a game oftic-tac-toe. 这道题让我们判断玩家是否能赢井字棋游戏, ...

随机推荐

  1. 关于华为服务器 双路E52620安装系统时遇到的问题

    一   准备好u盘启动盘,centos7的,用UltraISO制作启动盘 双击图标 进入 点击启动下的刻录磁盘映像 几分钟之后制作完场 二 :按装centos7的系统 1 首先将 u盘插到服务器上 2 ...

  2. 树(Tree)解题总结

    定义 树是一种抽象数据类型(ADT)或是实现这种抽象数据类型的数据结构,用来模拟具有树状结构性质的数据集合.它是由 n(n>0) 个有限节点组成一个具有层次关系的集合.. 二叉搜索树(Binar ...

  3. FileZilla Server FTP服务器失败

    使用Filezilla Server配置FTP服务器https://blog.csdn.net/chuyouyinghe/article/details/78998527 FileZilla Serv ...

  4. 聊一聊RocketMQ的注册中心NameServer

    前言 上次我们一起了解了RocketMQ的基本架构原理,那简单的回顾一下RocketMQ的架构组成. RocketMQ其实包含了四个核心部分,NameServer.Broker.Producer.Co ...

  5. Kubernetes-14:一文详解Pod、Node调度规则(亲和性、污点、容忍、固定节点)

    Kubernetes Pod调度说明 简介 Scheduler 是 Kubernetes 的调度器,主要任务是把定义的Pod分配到集群的节点上,听起来非常简单,但要考虑需要方面的问题: 公平:如何保证 ...

  6. VuePress初探(一)

    原文参考链接 手把手教你使用 VuePress 搭建个人博客 有阅读障碍的同学,可以跳过第一至四节,下载我写好的工具包: git clone https://github.com/zhangyunch ...

  7. Lua索引、伪索引、引用

    索引:堆栈的索引 伪索引:一个类似于索引,但是有着特殊存储的索引,使用方式和索引一样,看上去像在操作堆栈 引用:LUA_REGISTRYINDEX伪索引下的表的整数键

  8. [BUUOJ记录] [BJDCTF2020]Easy MD5

    各种关于md5的Bypass操作,都是基本操作,考察数组绕过.弱类型比较绕过以及md5($password,true) ByPass 1.利用md5($password,true)实现SQL注入 F1 ...

  9. 2 http

    response.write(string|buffer)可以调用0-n次 response.end(string|buffer) 方法.必须调用一次  response.setHeader('Con ...

  10. Kubernetes K8S在IPVS代理模式下Service服务的ClusterIP类型访问失败处理

    Kubernetes K8S使用IPVS代理模式,当Service的类型为ClusterIP时,如何处理访问service却不能访问后端pod的情况. 背景现象 Kubernetes K8S使用IPV ...