井字棋小游戏(C语言)
最近沉迷于《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语言)的更多相关文章
- [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 ...
- JavaScript写的一个带AI的井字棋
最近有一门课结束了,需要做一个井字棋的游戏,我用JavaScript写了一个.首先界面应该问题不大,用html稍微写一下就可以.主要是人机对弈时的ai算法,如何使电脑方聪明起来,是值得思考一下的.开始 ...
- [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 ...
- [HTML5实现人工智能]小游戏《井字棋》发布,据说IQ上200才能赢
一,什么是TicTacToe(井字棋) 本 游戏 为在下用lufylegend开发的第二款小游戏.此游戏是大家想必大家小时候都玩过,因为玩它很简单,只需要一张草稿纸和一只笔就能开始游戏,所以广受儿 ...
- 『HTML5实现人工智能』小游戏《井字棋》发布,据说IQ上200才能赢【算法&代码讲解+资源打包下载】
一,什么是TicTacToe(井字棋) 本游戏为在下用lufylegend开发的第二款小游戏.此游戏是大家想必大家小时候都玩过,因为玩它很简单,只需要一张草稿纸和一只笔就能开始游戏,所以广受儿童欢迎. ...
- Pascal小游戏 井字棋
一个很经典的井字棋游戏 Pascal源码Chaobs奉上 注意:1.有的FP版本不支持汉语,将会出现乱码.2.别想赢电脑了,平手不错了. 井字过三关: program TicTacToe; uses ...
- Java 小游戏 - 井字棋 v1.0 (初步完成) (2018.4.16更新)
井字棋游戏初步完成 实现功能:输入位置数据->打印棋盘->判断是否胜利->继续游戏/退出游戏 缺点:没有清屏函数 判断胜利方法太过无脑 package MYGAME; ...
- 井字棋游戏升级版 - TopTicTacToe项目 简介
一.游戏简介 井字棋是一款世界闻名的游戏,不用我说,你一定知道它的游戏规则. 这款游戏简单易学,玩起来很有意思,不过已经证明出这款游戏如果两个玩家都足够聪明的话, 是很容易无法分出胜负的,即我们得到的 ...
- [CareerCup] 17.2 Tic Tac Toe 井字棋游戏
17.2 Design an algorithm to figure out if someone has won a game oftic-tac-toe. 这道题让我们判断玩家是否能赢井字棋游戏, ...
随机推荐
- Element UI - DatePicker 自定义日期选择期间
<el-date-picker v-else v-model="searchForm.data_Selected" type="daterange" un ...
- Python开发的入门教程(八)-迭代
介绍 本文主要介绍Python中迭代的基本知识和使用 什么是迭代 在Python中,如果给定一个list或tuple,我们可以通过for循环来遍历这个list或tuple,这种遍历我们成为迭代(Ite ...
- Ingress-nginx 与 Nginx-ingress
一.概述 Ingress-nginx:它是由Kubernetes社区基于Nginx Web服务器开发的,并补充了一组用于实现额外功能的Lua插件,作为“官方”默认控制器支持当然最优. Github:h ...
- C#转PHP
官方主页 https://github.com/isukces/cs2php 快速开始 http://www.cs2php.com/how-to-begin.htm#.W2rBhC2B3mI 如何在V ...
- Mysql表,列,库的增删查改
下面是我总结的一些基础的sql知识,主要是为了以后更好的查阅和帮助其他初学的人,同时记录自己的成长,还写了一点稍有难度的sql面试题级别的题目,好了废话不多说,见真题... #创建数据库 CREATE ...
- Web最最基础
web 网站网页一个网站是由多个网页组成的一个网页=网页元素(文字.图片.超链接.文本框.按钮.下拉框ext.) +样式+用户交互 一个网页=(网页元素)html+(样式)CSS+(用户交互)Java ...
- 利用css3 transform实现一个时钟
transform:rotate(1deg) <!DOCTYPE html> <html lang="en"> <head> <meta ...
- Oracle WITH 语句 语法
With语句可以在查询中做成一个临时表/View,用意是在接下来的SQL中重用,而不需再写一遍. With Clause方法的优点: 增加了SQL的易读性,如果构造了多个子查询,结构会更清晰. 示例: ...
- 查看CentOs6.5/7的系统版本号
在centos6.5上用 [root@msg45 ~]# lsb_release -aLSB Version: :base-4.0-amd64:base-4.0-noarch:core-4.0- ...
- Sunday算法解决字符串匹配问题
概述 提起字符串匹配可能更多人会想到KMP算法,该算法时间复杂度为O(m+n),而且也是我们在学习数据结构过程中最早接触到的比较好的算法.但KMP算法需要在模式字符串有关联的情况下,也即模式字符串前后 ...