【C语言项目】三子棋

文章目录

项目思路

  1. 分文件进行创建
  2. 进入游戏前的目录
  3. 画出棋盘
  4. 玩家落子
  5. 电脑落子
  6. 输赢判断

接下来,我们分步骤进行详细的解释说明。

一、分文件进行创建

在具体的项目实施中,我们需要分成不同的文件进行创建和书写,以此来保证项目的模块化。

那么在三子棋的实际书写中,

  • 源文件:
    • 测试游戏用的代码文件用test.c或者日期.c来作为文件名
    • 游戏实现的底层代码用game.c文件作为文件名
  • 头文件
    • 游戏实现中使用的各个函数的声明 ,以及包含的其他库函数的头文件 需要写在头文件game.h文件里
      如图所示:

二、进入游戏前的目录

2.1 目录的功能:

  1. 在游戏开始时,给玩家视觉上的帮助和提示
  2. 让玩家可以选择进入游戏或者退出游戏
  3. 将游戏形成一个可以不断重玩的循环
    接下来,我们分步骤进行书写:

2.2 目录界面:

c 复制代码
void menu()
{
	printf("***********************\n");
	printf("*****1、开始游戏*******\n");
	printf("*****0、退出游戏*******\n");
	printf("***********************\n");
}

接着,在main函数里面进行调用:

c 复制代码
int main()
{
	menu();
	return 0;
}

这样目录的表面就写好了,接下来需要写玩家选择进入游戏和退出游戏的功能了

2.3 选择进入或退出游戏

选择功能的逻辑:1进入游戏,0退出游戏。

显然,必备的库函数有scanf,switch、case和default。为了让游戏的体验更加良好,可以再加一个printf增加视觉上的游玩帮助,用户友好。

那么根据刚刚的逻辑,可以写出如下的选择结构,其中game函数虽然还没有写出来,但是整体的框架可以先确定下来,之后再往里慢慢写内容。

c 复制代码
int main()
{
	int input = 0;
	menu();
	printf("请选择:--->");
	scanf("%d", &input);
	switch (input)
	{
	case 1:
		game();
		break;
	case 0:
		printf("退出游戏\n");
		break;
	default:
		printf("非法输入!请重试\n");
		break;
	}
	return 0;
}

2.4 多次重玩功能

多次重玩功能需要一个循环结构。由于开游戏的时候菜单页面必定会打印,所以菜单页面至少会运行一次,故可以使用 do while 循环结构

使用这个结构的同时,判断停止的条件就可以直接填写输入项,因为输入0是退出,而while为非0数字运行,所以刚好可以填写输入项,逻辑自洽。

c 复制代码
int main()
{
	int input = 0;
	do
	{
		menu();
		printf("请选择:--->");
		scanf("%d", &input);
		switch (input)
		{
		case 1:
			game();
			break;
		case 0:
			printf("退出游戏\n");
			break;
		default:
			printf("非法输入!请重试\n");
			break;
		}
	}
	while (input);
	return 0;
}

三、画出棋盘

3.1 写出棋子

在画棋盘的框架之前,需要找一个容器把棋子容纳进去,而3x3的棋盘,很明显用二维数组来进行盛放最为合适。

故可以写一个二维数组,当做棋盘,下棋就下在二维数组里面。

c 复制代码
	char board[ROW][COL] = { 0 };

直接写到void game函数里面就行了。

3.2 初始化棋盘

棋盘应该是全部空的,而不是初始化那样全部是0,所以可以写一个函数把数组里面的数据全部初始化成空格。

逻辑:遍历数组并赋值

代码示范:

c 复制代码
int i,j = 0 ;
void InitBoard(char board[ROW][COL], int row, int col)
{
	for (i = 0; i < row; i++)
	{
		for (j = 0; j < col; j++)
		{
			board[i][j] = ' ';
		}
	}
}

3.2 画出棋盘的框架

棋盘虽然也可以不画,直接9个字符位置下棋,但是太不美观,所以可以画一个美观一些的棋盘。

参考已经画出来的:

这个棋盘显然就比9个字符位美观多了,接下来就分步骤拆解它的输出:

第一行看起来是三个空格一个竖杠,但实际上,这里需要打印的不仅是棋盘的线,还要打印数组里面的棋子。

同时,代码不能写死,可以在game.h里面定义一个ROW(行)COL(列),这样的话,想要十乘十的棋盘,直接在game.h里面改数字就可以直接改全部的行列了。

game.h内:

c 复制代码
#define ROW 3
#define COL 3

这样,接下来的棋盘打印就可以用ROW和COL代替原来的3了。

3.3 代码实现

为了打印棋盘这个功能,我们需要声明并定义一个函数,使用函数进行各个不同模块的功能实现是在项目中十分有必要的。

game.h内:

接受的数据:棋盘数组、行、列

c 复制代码
void DisplayBoard(char board[ROW][COL], int row, int col);

而函数的定义则放在game.c内:

c 复制代码
void DisplayBoard(char board[ROW][COL], int row, int col)
{
	for (i = 0; i < row; i++)
	{
		//打印数据行
		for (j = 0; j < col; j++)//使用for循环打印每一格的数据
		{
			printf(" %c ", board[i][j]);
			if (j < COL - 1)//因为棋盘边缘没有边界线,所以少打印一个"|"
				printf("|");
		}
		printf("\n");//这里的换行需要留意别漏了
		
		//打印分割线行
		if (i < row - 1) {
			for (j = 0; j < row; j++)//使用for循环打印每一格的分割线
			{
				printf("---");
				if (j < row - 1)//打印"|"
					printf("|");//同理
			}
		}
		printf("\n");//这里的换行需要留意别漏了
	}
}

四、玩家落子

4.1 落子逻辑

玩家落子的逻辑是输入几行几列的坐标,然后棋盘在对应的位置上出现一个符号,相当于是落子。

4.2具体情况分类讨论

  • 当玩家落子正确
    • 将" * "放入数组
  • 当玩家落子不在棋盘内
    • 打印提示,让玩家重新输入
  • 当玩家落子时棋盘已经有子
    • 打印提示,让玩家重新输入

这三种情况需要不同的代码来实现

  1. 判断是否在棋盘内,可以用坐标是否在棋盘的范围内的if语句判断
  2. 判断是否已经落子,可以用数组里的数据是否为空格来判断,若不是空格,即有子,不能下
  3. 如果都可以,就放入数组一个 * 号,然后break跳出循环

经过分析,不难发现,这里的循环是直到下到正确的棋才会跳出循环,所以只需使用while循环,条件里填1或者其他非0数字,就可以一直循环了。

4.3代码示范

头文件中声明函数

c 复制代码
void PlayerMove(char board[ROW][COL], int row, int col);

源文件中定义函数

c 复制代码
void PlayerMove(char board[ROW][COL], int row, int col)
{
	printf("请输入棋子坐标:");
	while (1)
	{
		scanf("%d %d", &i, &j);//输入坐标
		if (i > 0 && i<= row && j>0 && j <= col)//判断是否在棋盘内
		{
			if (board[i - 1][j - 1] == ' ')//判断是否有子
			{
				board[i - 1][j - 1] = '*';
				break;
			}
			else
				printf("已经落子,请重新输入\n");
		}
		else
			printf("非法输入\n");
	}
}

五、电脑落子

5.1 电脑落子的逻辑

首先电脑落子是需要一个随机性的,那么就可以使用srand和rand函数(伪随机数),加上时间戳构成一个真随机数,再利用这个真随机数取一个模,就可以在棋盘里下棋了。

5.2分类讨论

至于实际上的分类逻辑,和玩家下棋不太一样,只有两种情况:

  • 当电脑落子正确
    • 将" # "放入数组
  • 当电脑落子时棋盘已经有子
    • 电脑重新落子

5.3 代码示范

函数声明

c 复制代码
void ComputerMove(char board[ROW][COL], int row, int col);

函数定义

c 复制代码
void ComputerMove(char board[ROW][COL], int row, int col)
{
	while (1) //和玩家落子同理,不下对棋就继续下,故while(1)
	{
		i = rand() % row;//行的随机数取模
		j = rand() % col;//列的随机数取模
		if (board[i][j] == ' ')//判断是否是空位
		{
			board[i][j] = '#';//落子
			break;//跳出循环
		}
	}
}

这里注意rand需要和srand函数配合使用

在main函数中:

c 复制代码
	srand((unsigned int)time(NULL));

在头文件中:

c 复制代码
#include <stdio.h> //printf和scanf函数需要
#include <stdlib.h> //随机数需要用
#include <time.h> //时间戳需要用

六、输赢判断

输赢判断这里,由于规则是三字成线,且存在平局的情况,故需要分类讨论。

6.1 分类讨论

  1. 行三字成线
  2. 列三字成线
  3. 对角线三字成线
  4. 平局

6.2 行和列的三字成线

  1. 直接判断第一个棋子和第二个棋子是否相等,再并上第二个棋子与第三个棋子是否相等。
  2. 在判断相等的同时,需要判断是否是空格,如果是空格那就是没人赢。所以需要并上一个不等于空格的条件。

6.3 对角线的三字成线

  1. 同行列的判断条件,只是数位坐标需要换成对角线的。

6.4平局

  1. 当棋盘落满,且没有人胜出的时候,就可以判定为平局。
  2. 棋盘是否落满的逻辑:遍历二维数组,并判断是否为空格。如果每一个位置都不是空格,那就是落满了,返回1,如果有任何一个位置是空格,立马返回0。

6.5 代码实现:

6.5.1 判断输赢

c 复制代码
char IsWin(char board[ROW][COL], int row, int col)
{
	//行
	for (i = 0; i < row; i++)
		if (board[i][0] == board[i][1] && board[i][1] == board[i][2]&& board[i][0]!=' ')
			return board[i][0];
	//列
	for (i = 0; i < col; i++)
		if (board[0][i] == board[1][i] && board[1][i] == board[2][i] && board[0][i] != ' ')
			return board[0][i];
	//对角线
	if (board[0][0] == board[1][1] && board[1][1] == board[2][2] && board[0][0] != ' ')
		return board[0][0];
	if (board[0][2] == board[1][1] && board[1][1] == board[2][0] && board[0][2] != ' ')
		return board[0][2];
	//平局
	if (IsFull(board,row,col) == 1)
		return 'D';
	return 'C';
}

6.5.2 判断棋盘是否满

c 复制代码
int IsFull(char board[ROW][COL], int row, int col)
{
	for (i = 0; i < row; i++)
	{
		for (j = 0; j < col; j++)
		{
			if (board[i][j] == ' ')//遍历并判断是否是空格
				return 0;
		}
	}
	return 1;
}

注:以上的函数都需要在头文件中进行声明,声明格式同上

七、完整代码示范(无注释)

7.1 test.c

c 复制代码
#define _CRT_SECURE_NO_WARNINGS 1
#include "game.h"
void menu()
{
	printf("***********************\n");
	printf("*****1、开始游戏*******\n");
	printf("*****0、退出游戏*******\n");
	printf("***********************\n");
}

void game()
{
	char ret = 0;
	char board[ROW][COL] = { 0 };
	InitBoard(board,ROW,COL);
	DisplayBoard(board,ROW,COL);
	while (1)
	{
		PlayerMove(board, ROW, COL);
		DisplayBoard(board, ROW, COL);
		ret=IsWin(board, ROW, COL);
		if (ret != 'C')
			break;
		ComputerMove(board, ROW, COL);
		DisplayBoard(board, ROW, COL);
		ret = IsWin(board, ROW, COL);
		if (ret != 'C')
			break;
	}
	//平局Draw
	if (ret == '#')
		printf("电脑赢\n");
	else if (ret == '*')
		printf("玩家赢\n");
	else if (ret == 'D')
		printf("平局\n");
	else
		printf("程序出错\n");
}

int main()
{
	int input = 0;
	srand((unsigned int)time(NULL));
	do
	{
		menu();
		printf("请选择:--->");
		scanf("%d", &input);
		switch (input)
		{
		case 1:
			game();
			break;
		case 0:
			printf("退出游戏\n");
			break;
		default:
			printf("非法输入!请重试\n");
			break;
		}
	}
	while (input);
	return 0;
}

7.2 game.h

c 复制代码
#pragma once
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#define ROW 3
#define COL 3
void InitBoard(char board[ROW][COL], int row, int col);
void DisplayBoard(char board[ROW][COL], int row, int col);
void PlayerMove(char board[ROW][COL], int row, int col);
void ComputerMove(char board[ROW][COL], int row, int col);
char IsWin(char board[ROW][COL], int row, int col);
int IsFull(char board[ROW][COL], int row, int col);

7.3 game.c

c 复制代码
#define _CRT_SECURE_NO_WARNINGS 1
#include "game.h"
//初始化棋盘为空格
int i, j = 0;
void InitBoard(char board[ROW][COL], int row, int col)
{
	for (i = 0; i < row; i++)
	{
		for (j = 0; j < col; j++)
		{
			board[i][j] = ' ';
		}
	}
}

//打印棋盘
void DisplayBoard(char board[ROW][COL], int row, int col)
{
	for (i = 0; i < row; i++)
	{
		//打印数据行
		for (j = 0; j < col; j++)
		{
			printf(" %c ", board[i][j]);
			if (j < COL - 1)
				printf("|");
		}
		printf("\n");

		//打印分割线行
		if (i < row - 1) {
			for (j = 0; j < row; j++)
			{
				printf("---");
				if (j < row - 1)
					printf("|");
			}
		}

		printf("\n");
	}

}

//玩家下棋
void PlayerMove(char board[ROW][COL], int row, int col)
{
	printf("请输入棋子坐标:");
	while (1)
	{
		scanf("%d %d", &i, &j);
		if (i > 0 && i<= row && j>0 && j <= col)
		{
			if (board[i - 1][j - 1] == ' ')
			{
				board[i - 1][j - 1] = '*';
				break;
			}
			else
				printf("已经落子,请重新输入\n");
		}
		else
			printf("非法输入\n");
	}
}

//电脑下棋
void ComputerMove(char board[ROW][COL], int row, int col)
{
	while (1)
	{
		i = rand() % row;
		j = rand() % col;
		if (board[i][j] == ' ')
		{
			board[i][j] = '#';
			break;
		}
	}
}

int IsFull(char board[ROW][COL], int row, int col)
{
	for (i = 0; i < row; i++)
	{
		for (j = 0; j < col; j++)
		{
			if (board[i][j] == ' ')
				return 0;
		}
	}
	return 1;
}


char IsWin(char board[ROW][COL], int row, int col)
{
	//行
	for (i = 0; i < row; i++)
		if (board[i][0] == board[i][1] && board[i][1] == board[i][2]&& board[i][0]!=' ')
			return board[i][0];
	//列
	for (i = 0; i < col; i++)
		if (board[0][i] == board[1][i] && board[1][i] == board[2][i] && board[0][i] != ' ')
			return board[0][i];
	//对角线
	if (board[0][0] == board[1][1] && board[1][1] == board[2][2] && board[0][0] != ' ')
		return board[0][0];
	if (board[0][2] == board[1][1] && board[1][1] == board[2][0] && board[0][2] != ' ')
		return board[0][2];
	//平局
	if (IsFull(board,row,col) == 1)
		return 'D';
	return 'C';
}

7.4 运行图片(示例)

写在最后

如果本文对您有帮助,可不可以给我一个小小的点赞呀❤~您的支持是我最大的动力。

博主小白一枚,才疏学浅,难免有所纰漏,欢迎大家讨论和提出问题,博主一定第一时间改正。

谢谢观看嘿嘿(๑•̀ㅂ•́)و✧~!

相关推荐
Am心若依旧40930 分钟前
[c++11(二)]Lambda表达式和Function包装器及bind函数
开发语言·c++
明月看潮生32 分钟前
青少年编程与数学 02-004 Go语言Web编程 20课题、单元测试
开发语言·青少年编程·单元测试·编程与数学·goweb
大G哥41 分钟前
java提高正则处理效率
java·开发语言
stm 学习ing42 分钟前
HDLBits训练5
c语言·fpga开发·fpga·eda·hdlbits·pld·hdl语言
VBA63371 小时前
VBA技术资料MF243:利用第三方软件复制PDF数据到EXCEL
开发语言
轩辰~1 小时前
网络协议入门
linux·服务器·开发语言·网络·arm开发·c++·网络协议
小_太_阳1 小时前
Scala_【1】概述
开发语言·后端·scala·intellij-idea
向宇it1 小时前
【从零开始入门unity游戏开发之——unity篇02】unity6基础入门——软件下载安装、Unity Hub配置、安装unity编辑器、许可证管理
开发语言·unity·c#·编辑器·游戏引擎
古希腊掌管学习的神2 小时前
[LeetCode-Python版]相向双指针——611. 有效三角形的个数
开发语言·python·leetcode
赵钰老师2 小时前
【R语言遥感技术】“R+遥感”的水环境综合评价方法
开发语言·数据分析·r语言