C语言实现五子棋小游戏

C语言实现五子棋小游戏三子棋,五子棋,无论多少子棋,其原理都是一样的。下面我用五子棋为例讲解用C语言多文件编程实现五子棋。设计电脑和玩家两个作为下棋的两方,用键盘输入作为玩家的游戏操作。1.效果图:程序总的构架:我们只要输入坐标就可以和电脑对弈了。电脑的棋子用‘0’表示,玩家的棋子用‘x’表示。2.打印菜单可以根据自己的爱好设计各种风格的…

大家好,又见面了,我是你们的朋友全栈君。

       三子棋,五子棋,无论多少子棋,其原理都是一样的。下面我用五子棋为例讲解用C语言多文件编程实现五子棋。

       设计电脑和玩家两个作为下棋的两方,用键盘输入作为玩家的游戏操作。

1.效果图:

程序总的构架:

C语言实现五子棋小游戏

我们只要输入坐标就可以和电脑对弈了。

电脑的棋子用 ‘0’ 表示,玩家的棋子用 ‘x’ 表示。

C语言实现五子棋小游戏

C语言实现五子棋小游戏

 

2.打印菜单

可以根据自己的爱好设计各种风格的菜单,自己自然赏心悦目,让自己的游戏更加美观。

// 菜单
menu()
{
	printf("\n");
	printf("*******************************\n");
	printf("****  欢迎来到五子棋游戏!  ****\n");
	printf("****      1.进入游戏       ****\n");
	printf("****      0.退出游戏       ****\n");
	printf("*******************************\n");
}

 

3.初始化棋盘

ROW, COL 分别表示棋盘的宽度和高度(即是棋盘的 x 和 y)。

一开始先把棋盘初始化为 ‘  ‘  (空格)。

void InitBoard(char board[ROW][COL], int row, int col)
{
	int i = 0;
	int j = 0;

	for (i = 0; i < row; i++)
	{
		for (j = 0; j < col; j++)
		{
			board[i][j] = ' ';
		}
	}
}

4.打印棋盘

用竖线和横线把棋盘封装起来,在棋盘外标上坐标的位置提示,以方便玩家。

//打印棋盘 
void DisplayBoard(char board[ROW][COL], int row, int col)
{
	int i = 0;
	int j = 0;

	for (i = 0; i < row; i++)
	{
		printf("  %d ", i+1); //打印棋盘 x 轴坐标提示
	}
	printf("\n");

	for (j = 0; j < col; j++)
	{
		printf("---|"); //打印第一行棋盘
	}
	printf("\n");

	for (i = 0; i < row; i++)
	{
		for (j = 0; j < col; j++)
		{
			printf(" %c |", board[i][j]); //打印竖标
		}
		printf(" %d ", i+1); //打印棋盘 y 轴坐标提示
		printf("\n");

		for (j = 0; j < col; j++)
		{
			printf("---|"); //打印横标
		}
		printf("\n");
	}
}

5.电脑下棋

我是利用函数 strand() 函数和 rand() 函数让电脑在棋盘上空的地方随机下棋,用一个循环判断就可以实现。也可以设计电脑让电脑更加“聪明”,能够判断玩家已经下好的棋,并能够做出阻止(我没有做这一步,如果有兴趣的话,可以自己再研究一下哦 ^_^)。

//电脑下棋
void ComputerMove(char board[ROW][COL], int row, int col)
{
	int x = 0;
	int y = 0;
	printf("电脑走:>\n");

	while (1)
	{
		x = rand() % row;
		y = rand() % col;

		if (board[x][y] == ' ')
		{
			board[x][y] = '0';
			break;
		}
		else
		{
			continue;
		}
	}
}

6.玩家下棋

从键盘上输入下棋坐标,判断坐标正好是棋盘上空的地方,则成功下棋,若在棋盘上但非空位置,提示玩家该坐标已经被占用,要重新输入,若不在棋盘上,提示玩家该坐标非法,要重新输入。

//玩家下棋
void PlayerMove(char board[ROW][COL], int row, int col)
{
	int x = 0;
	int y = 0;

	printf("玩家走:>\n");
	printf("请输入坐标(%d,%d): >",row,col);

	while (1)
	{
		scanf("%d %d", &x, &y);
		if (x >= 1 && x <= row && y >= 1 && y <= col)
		{
			// 成功下棋
			if (board[x-1][y-1] == ' ')
			{
				board[x-1][y-1] = 'x';
				break;
			}
			// 输入坐标已被占用
			else
			{
				printf("该坐标已经被占用\n");
				printf("请重新输入:>");
				continue;
			}
		}
		// 输入坐标非法判断
		else
		{
			printf("坐标非法\n");
			printf("请重新输入:>");
			continue;
		}
	}
}

7.判断平局

不是平局返回 0 ,平局返回 1 。只要棋盘上有空位置则返回 0 表示不是平局。

//判断平局
static int IsFull(char board[ROW][COL], int row, int col)
{
	int i = 0;
	int j = 0;

	for (i = 0; i < row; i++)
	{
		for (j = 0; j < col; j++)
		{
			if (board[i][j] == ' ')
			{
				return 0;
			}
		}
	}

	//棋盘没有空位置了还没有判断出输赢,则平局
	return 1;
}

8.判断输赢

两方谁先把五颗棋子连成一线,就是赢家。判断输赢函数是一个重要的模块,其代码如下:

//判断输赢
char IsWin(char board[ROW][COL], int row, int col)
{
	int i = 0;
	int j = 0;

	// 横线上五子连成一线,赢家产生
	for (i = 0; i < row; i++)
	{
		for (j = 0; j < col - 4; j++)
		{
			if (board[i][j] == board[i][j + 1]
				&& board[i][j + 1] == board[i][j + 2]
				&& board[i][j + 2] == board[i][j + 3]
				&& board[i][j + 3] == board[i][j + 4]
				&& board[i][j] != ' ')
			{
				return board[i][j];
			}
		}
	}

	// 竖线上五子连成一线,赢家产生
	for (j = 0; j < col; j++)
	{
		for (i = 0; i < row - 4; i++)
		{
			if (board[i][j] == board[i+1][j]
				&& board[i+1][j] == board[i+2][j] 
				&&board[i+2][j] == board[i+3][j]
				&& board[i+3][j] == board[i+4][j] 
				&& board[i][j] != ' ')
			{
				return board[i][j];
			}
		}
	}

	// 斜线上五子连成一线,赢家产生
	for (i = 0; i < row - 4; i++)
	{
		if (board[i][i] == board[i+1][i+1]
			&& board[i+1][i+1] == board[i+2][i+2]
			&& board[i + 2][i + 2] == board[i + 3][i + 3]
			&& board[i + 3][i + 3] == board[i + 4][i + 4]
			&& board[i][i] != ' ')
		{
			return board[i][i];
		}

		if (board[i][i+4] == board[i+1][i+3]
			&& board[i+1][i+3] == board[i+2][i+2]
			&& board[i + 2][i + 2] == board[i + 3][i + 1]
			&& board[i + 3][i + 1] == board[i + 4][i]
			&& board[i][i + 4] != ' ')
		{
			return board[i][i+4];
		}
	}

	//游戏平局
	if (IsFull(board, row, col))
	{
		return 'p';
	}

	//游戏结束
	return ' ';

}

9.游戏执行

以上就是我们要实现扫雷的模块,要想把这些模块整合起来运行,就需要一个游戏执行函数来调用这些模块,定义个game()函数实现,代码如下:

// 游戏开始执行
void game()
{
	int ret = 0;
	char board[ROW][COL] = { 0 };
	InitBoard(board, ROW, COL);

	// 下棋
	while (1)
	{
		ComputerMove(board, ROW, COL); //电脑走
		ret = IsWin(board, ROW, COL);
		if (ret != ' ')
		{
			break;
		}
		system("CLS"); //清屏,优化界面
		DisplayBoard(board, ROW, COL); //打印棋盘
		printf("\n");

		PlayerMove(board, ROW, COL); //玩家走
		ret = IsWin(board, ROW, COL);
		if (ret != ' ')
		{
			break;
		}
		DisplayBoard(board, ROW, COL); //打印棋盘
		printf("\n");
	}

	// 判断输赢或平局
	if (ret == 'p')
	{
		printf("平局\n"); 
		DisplayBoard(board, ROW, COL); //打印棋盘
	}
	else if (ret == 'x')
	{
		printf("玩家赢\n");
		DisplayBoard(board, ROW, COL); //打印棋盘
	}
	else if (ret == '0')
	{
		printf("电脑赢\n");
		DisplayBoard(board, ROW, COL); //打印棋盘
	}
}

10.头文件

在头文件 game.h 中声明各种函数,并将头文件 game.h 放在 main.c 文件中。

#ifndef __GAME_H__
#define __GAME_H__

# define _CRT_SECURE_NO_WARNINGS

#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <string.h>

#define ROW 10
#define COL 10

void InitBoard(char board[ROW][COL], int row, int col);
void DisplayBoard(char board[ROW][COL], int row, int col);
void ComputerMove(char board[ROW][COL], int row, int col);
void PlayerMove(char board[ROW][COL], int row, int col);
char IsWin(char board[ROW][COL], int row, int col);

#endif  __GAME_H__

11.测试

可以将棋盘的 x 和 y 轴长度更改多个值进行测试,已确保代码的稳定性。

void test()
{
	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");
		}
	} while (input);
}

12.附:game.c 的源码

#include "game.h"

// 菜单
menu()
{
	printf("\n");
	printf("*******************************\n");
	printf("****  欢迎来到五子棋游戏! ****\n");
	printf("****      1.进入游戏       ****\n");
	printf("****      0.退出游戏       ****\n");
	printf("*******************************\n");
}

//初始化
void InitBoard(char board[ROW][COL], int row, int col)
{
	int i = 0;
	int j = 0;

	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)
{
	int i = 0;
	int j = 0;

	for (i = 0; i < row; i++)
	{
		printf("  %d ", i+1); //打印棋盘 x 轴坐标提示
	}
	printf("\n");

	for (j = 0; j < col; j++)
	{
		printf("---|"); //打印第一行棋盘
	}
	printf("\n");

	for (i = 0; i < row; i++)
	{
		for (j = 0; j < col; j++)
		{
			printf(" %c |", board[i][j]); //打印竖标
		}
		printf(" %d ", i+1); //打印棋盘 y 轴坐标提示
		printf("\n");

		for (j = 0; j < col; j++)
		{
			printf("---|"); //打印横标
		}
		printf("\n");
	}
}

//电脑下棋
void ComputerMove(char board[ROW][COL], int row, int col)
{
	int x = 0;
	int y = 0;
	printf("电脑走:>\n");

	while (1)
	{
		x = rand() % row;
		y = rand() % col;

		if (board[x][y] == ' ')
		{
			board[x][y] = '0';
			break;
		}
		else
		{
			continue;
		}
	}
}

//玩家下棋
void PlayerMove(char board[ROW][COL], int row, int col)
{
	int x = 0;
	int y = 0;

	printf("玩家走:>\n");
	printf("请输入坐标(%d,%d): >",row,col);

	while (1)
	{
		scanf("%d %d", &x, &y);
		if (x >= 1 && x <= row && y >= 1 && y <= col)
		{
			// 成功下棋
			if (board[x-1][y-1] == ' ')
			{
				board[x-1][y-1] = 'x';
				break;
			}
			// 输入坐标已被占用
			else
			{
				printf("该坐标已经被占用\n");
				printf("请重新输入:>");
				continue;
			}
		}
		// 输入坐标非法判断
		else
		{
			printf("坐标非法\n");
			printf("请重新输入:>");
			continue;
		}
	}
}

//判断平局
static int IsFull(char board[ROW][COL], int row, int col)
{
	int i = 0;
	int j = 0;

	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)
{
	int i = 0;
	int j = 0;

	// 横线上五子连成一线,赢家产生
	for (i = 0; i < row; i++)
	{
		for (j = 0; j < col - 4; j++)
		{
			if (board[i][j] == board[i][j + 1]
				&& board[i][j + 1] == board[i][j + 2]
				&& board[i][j + 2] == board[i][j + 3]
				&& board[i][j + 3] == board[i][j + 4]
				&& board[i][j] != ' ')
			{
				return board[i][j];
			}
		}
	}

	// 竖线上五子连成一线,赢家产生
	for (j = 0; j < col; j++)
	{
		for (i = 0; i < row - 4; i++)
		{
			if (board[i][j] == board[i+1][j]
				&& board[i+1][j] == board[i+2][j] 
				&&board[i+2][j] == board[i+3][j]
				&& board[i+3][j] == board[i+4][j] 
				&& board[i][j] != ' ')
			{
				return board[i][j];
			}
		}
	}

	// 斜线上五子连成一线,赢家产生
	for (i = 0; i < row - 4; i++)
	{
		if (board[i][i] == board[i+1][i+1]
			&& board[i+1][i+1] == board[i+2][i+2]
			&& board[i + 2][i + 2] == board[i + 3][i + 3]
			&& board[i + 3][i + 3] == board[i + 4][i + 4]
			&& board[i][i] != ' ')
		{
			return board[i][i];
		}

		if (board[i][i+4] == board[i+1][i+3]
			&& board[i+1][i+3] == board[i+2][i+2]
			&& board[i + 2][i + 2] == board[i + 3][i + 1]
			&& board[i + 3][i + 1] == board[i + 4][i]
			&& board[i][i + 4] != ' ')
		{
			return board[i][i+4];
		}
	}

	//游戏平局
	if (IsFull(board, row, col))
	{
		return 'p';
	}

	//游戏结束
	return ' ';

}

 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/144642.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • C# 事务之SqlTransaction

    C# 事务之SqlTransactionC#事务之SqlTransactionprivatestaticvoidExecute(stringconnectionString){using(SqlConnectionconnection=newSqlConnection(connectionString)){…

  • 多层感知机理解(多层感知机原理)

    多层感知器(Muti-LayerPercetron)和卷积网络(ConvolutionalNeuralNetwork)。这两种网络都属于前馈型网络(Feedforwardnetwork),其中多层感知器(MLP)是最简单也是最常见的一种神经网络结构,它是所有其他神经网络结构的基础,好在我对神经网络的了解是从卷积神经网络开始的,对基本的原理和模型已经有了了解,所以学习起来相对容易,先看多层感知机的模型:1.网络的连接及输出:1.最左边就是我们的输入层了,算是网络的第0层,通常是一个向量x:

  • QQ自动强制加好友代码

    强行聊天的代码:tencent://Message/?Uin=919433667&amp;websiteName=www.oicqzone.com&amp;Menu=yes强行加好友的代码:tencent://AddContact/fromId=45&amp;fromSubId=1&amp;subcmd=all&amp;uin=574201314&amp;fuin=919433667&amp;w…

  • docker开启2375端口[通俗易懂]

    docker开启2375端口[通俗易懂]Docker开启RemoteAPI访问2375端口-hongdada-博客园https://www.cnblogs.com/hongdada/p/11512901.htmldocker开启2375端口,提供外部访问docker,idea连接服务器docker_霓虹深处-CSDN博客_idea连接docker2376端口https://blog.csdn.net/qq_36850813/article/details/89924207…

  • visio产品密钥2016_visio产品密钥2003

    visio产品密钥2016_visio产品密钥2003http://blog.sina.com.cn/s/blog_4e0869690100z9m3.htmlVisioPremium2010VOL版:=========================F…

  • 云铺购代刷网系统全开源可运营程序搭建「建议收藏」

    云铺购代刷网系统全开源可运营程序搭建「建议收藏」云铺购最新代刷网系统无后门全开源可运营版本控制端功能支持一键通秒搭建代刷网站点,一键新增修改站点版本,支持QQ一键通登录自主添加站点域名管理站点,可配置后台安全访问域名白名单IP(实时保护)控制端支持一键备份旗下所有站点数据,共享数据版大大减少服务器压力主站点功能前后台支持QQ一键通登录,前台风格8套内页风格3套,免密支付,订单代付自定义网站公告导航,等级配置,邮箱配置,密匙配置,站点一键通装修支持一键通秒对接云铺购系统,玖伍系统,亿乐系统,各大卡盟系统,网商系统等对接商

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号