C语言十字消除游戏!(超简单详细)详细思路+源码分享
liebian365 2024-10-26 13:00 30 浏览 0 评论
十字消除,休闲小游戏,在规定时间内,只要是十字线能连接到的相同颜色的方块,就能获得相应的得分,如果点击后没有能够消除的方块会扣除时间,是一款益智小游戏。
我们将编写十字消除游戏,用户点击空白方块,沿其上下左右方向寻找第一个彩色方块,如果有两个或两个以上颜色一致,就将其消除。在进度条时间结束前消除足够的方块,可以进入下一关,效果如图所示:
绘制过程
首先实现随机颜色方块的表示与绘制,鼠标点击与十字消除算法;然后绘制了提示框,绘制倒计时进度条;接着进行了得分计算、胜负判断、多关卡功能的开发;学习了地址与指针的概念,并利用地址传递使得程序更加模块化;最后学习了指针和数组的知识,应用动态数组实现了游戏尺寸的动态大小调整
在规定时间内,只要是十字线能连接到的相同颜色的方块,就能获得相应的得分,如果点击后没有能够消除的方块会扣除时间。
操作方法
鼠标点击两个或多个颜色相同方块的十字线的中心,便能成功消除方块。
注意是点击空白格子而不是彩色方块哦,点错会扣时间,点左下角的[显示图标提示]可切换到图标模式
代码示例:
#include <graphics.h>
#include <conio.h>
#include <stdio.h>
#include <time.h>
#include <stdlib.h>
# define BlockSize 40 // 小方块的长宽大小
# define ColorTypeNum 9 // 除了空白方块外,其他方块的颜色的个数
struct Block // 小方块结构体
{
int x,y; // x y坐标
int colorId; // 对应颜色的下标
int i,j; // 小方块在二维数组中的i j下标
};
// 全局变量
int RowNum; // 游戏画面一共RowNum行小方块
int ColNum; // 游戏画面一共ColNum列小方块
Block **blocks = NULL; // 动态二维数组指针,存储所有方块数据
COLORREF colors[ColorTypeNum+1]; // 颜色数组,小方块可能的几种颜色
int score; // 得分数,也就是消去的方块的个数
float maxTime; // 这一关游戏的总时长
float totalTime; // 减去扣分项后的游戏总时长
float remainTime; // 剩余时间
clock_t start, finish; // 用于计时的变量
int level = 1; // 当前关卡序号
int noZeroBlockNum; // 非空白区域的砖块的个数
void drawBlockHint(int i,int j,COLORREF color,int isfill) // 绘制出一个提示线框出来
{
setlinecolor(color);
setfillcolor(color);
if (isfill==1) // 鼠标点击中的方块,画填充方块提示
fillrectangle(blocks[i][j].x,blocks[i][j].y,blocks[i][j].x+BlockSize,blocks[i][j].y+BlockSize);
if (isfill==0) // 上下左右四个方向找到的4个方块,画线框提示
rectangle(blocks[i][j].x,blocks[i][j].y,blocks[i][j].x+BlockSize,blocks[i][j].y+BlockSize);
}
void writeRecordFile(int recordScore) //保存最高分数据文件
{
FILE *fp;
fp = fopen(".\\gameRecord.dat","w");
fprintf(fp,"%d",recordScore);
fclose(fp);
}
int readRecordFile() //读取最高分数据文件
{
int recordScore;
FILE *fp;
fp = fopen(".\\gameRecord.dat","r");
// 如果打不开的话,就新建一个文件,其得分记录为0分
if (fp==NULL)
{
writeRecordFile(0);
return 0;
}
else // 能打开这个文件,就读取下最高分记录
{
fscanf(fp,"%d",&recordScore);
fclose(fp);
return recordScore;
}
}
void startup() // 初始化函数
{
int i,j;
start = clock(); // 记录当前运行时刻
if (level>1) // 如果不是第1关,则先清除二维数组内存,再重新开辟内存空间
{
for (i=0;i<RowNum;i++)
free(blocks[i]);
free(blocks);
}
// 根据是第几关,调整这一关对应的游戏画面的大小
RowNum = 12 + level/2; // 行数添加的慢一些,是一个长方形的形状
ColNum = 20 + level;
// 开辟动态二维数组
blocks = (Block **) malloc(RowNum*sizeof(Block *));
for (i=0;i<RowNum;i++)
blocks[i] = (Block *) malloc(ColNum*sizeof(Block));
maxTime = 200 + level*10; // 这一关游戏设定的总时长,每关时长+10秒
totalTime = maxTime; // 游戏总时长,每次出错,会扣10秒钟
int width = BlockSize*ColNum; // 设定游戏画面的大小
int height = BlockSize*(RowNum+3); // 最下面用来显示一些提示信息
initgraph(width,height);
setbkcolor(RGB(220,220,220));
setlinestyle(PS_SOLID,2);
cleardevice();
srand(time(0));
BeginBatchDraw(); // 开始批量绘制
score = 0; // 得分数,也就是消去的方块的个数
noZeroBlockNum = 0; // 非空白区域的砖块的个数
colors[0] = RGB(220,220,220); // 颜色数组第一种颜色为灰白色,表示空白小方块
for (i=1;i<ColorTypeNum+1;i++) // 其他几种颜色为彩色
colors[i] = HSVtoRGB((i-1)*40,0.6,0.8);
// 对blocks二维数组进行初始化
for (i=0;i<RowNum;i++)
{
for (j=0;j<ColNum;j++)
{
// 取随机数,1-6设为彩色色块,其他为空白色块,这样为空白色块的几率高一些
// 初始化时,空白色块多一些,符合游戏的设定
int t = rand()%(int(ColorTypeNum*1.5)); // 取随机数
if (t<ColorTypeNum+1)
blocks[i][j].colorId = t; // 小方块的颜色序号
else // 其他情况,都为空白颜色方块
blocks[i][j].colorId = 0; // 小方块的颜色序号
blocks[i][j].x = j*BlockSize; // 小方块左上角坐标
blocks[i][j].y = i*BlockSize; //
blocks[i][j].i = i; // 存储当前小方块在二维数组中的下标
blocks[i][j].j = j;
if (blocks[i][j].colorId != 0)
noZeroBlockNum++; // 统计随机产生的方块中,非零方块的个数
}
}
}
void show() // 绘制函数
{
cleardevice(); // 清屏
setlinecolor(RGB(255,255,255)); // 白色线条
int i,j;
for (i=0;i<RowNum;i++)
{
for (j=0;j<ColNum;j++)
{
// 以对应的颜色、坐标画出所有的小方块
setfillcolor(colors[blocks[i][j].colorId]);
fillrectangle(blocks[i][j].x,blocks[i][j].y,blocks[i][j].x+BlockSize,blocks[i][j].y+BlockSize);
}
}
// 根据剩余时间,绘制一个倒计时进度条,进度条按最大时间maxTime秒绘制
setlinecolor(RGB(255,0,0));
setfillcolor(RGB(255,0,0));
fillrectangle(0,BlockSize*(RowNum+0.2),remainTime*BlockSize*ColNum/maxTime,BlockSize*(RowNum+0.8));
// 输出得分文字
TCHAR s[80];
setbkmode(TRANSPARENT);
_stprintf(s, _T("%d"), score);
settextcolor(RGB(0,0,0));
settextstyle(22, 0, _T("宋体"));
outtextxy(BlockSize*(ColNum/2-0.1), BlockSize*(RowNum+0.2), s);
// 输出一些游戏提示信息
_stprintf(s, _T("点击空白方块,其十字区域有两个或以上相同颜色方块则消除;不能消除扣时间"));
outtextxy(BlockSize*(ColNum/15.0), BlockSize*(RowNum+1.2), s);
_stprintf(s, _T("目前第 %d 关,时间结束前得分达到 %d 可进入下一关"),level,int(noZeroBlockNum*0.9));
outtextxy(BlockSize*(ColNum/5.0), BlockSize*(RowNum+2.2), s);
FlushBatchDraw(); // 批量绘制
}
void updateWithoutInput() // 和输入无关的更新
{
// 倒计时减少
finish = clock(); // 当前时刻
// 从startup运行后,这一关游戏运行了多少秒
double duration = (double)(finish - start) / CLOCKS_PER_SEC;
remainTime = totalTime-duration; // 游戏剩余的时间
// 如果时间到了
if (remainTime<=0)
{
// 读一下文件记录,如果当前得分超过记录
if (score > readRecordFile())
{
// 更新下得分记录
writeRecordFile(score);
// 显示恭喜超过记录
show();
settextcolor(RGB(255,0,0));
settextstyle(100, 0, _T("黑体"));
outtextxy(BlockSize*(ColNum/30.0), BlockSize*(RowNum/3.0), _T("恭喜打破得分记录"));
FlushBatchDraw(); // 批量绘制
Sleep(2000);
}
if (score>=int(noZeroBlockNum*0.9))
{
level ++; // 如果得分达到要求,消除掉非空白方块数目的90%,关卡加1
}
startup(); // 调用初始化函数,重新开始游戏
return;
}
}
void updateWithInput() // 和输入有关的更新
{
if (remainTime<=0) // 时间到了,不要操作
return;
int i,j;
MOUSEMSG m;
if (MouseHit())
{
m = GetMouseMsg();
if(m.uMsg == WM_LBUTTONDOWN) // 当按下鼠标左键时
{
// 获得点击的小方块的下标
int clicked_i = int(m.y)/BlockSize;
int clicked_j = int(m.x)/BlockSize;
// 点击到下面提示部分了,不用处理,函数返回
if (clicked_i>=RowNum)
return;
// 如果当前点击的不是空白方块,不需要处理,返回
if (blocks[clicked_i][clicked_j].colorId!=0)
return;
show(); // 先显示其他方块,再绘制提示框,后绘制的在最前面
// 被点击到的空白方块,绘制下填充灰色方块提示框
drawBlockHint(clicked_i,clicked_j,RGB(100,100,100),1);
// 定义数组,存储上、下、左、右四个方向找到第一个不是空白的方块
Block fourBlocks[4] = {blocks[clicked_i][clicked_j]}; // 初始化为这个空白的点击的方块
int search; // 寻找下标
// 向上找
for (search=0;clicked_i-search>=0;search++)
{
if (blocks[clicked_i-search][clicked_j].colorId!=0) // 找到第一个颜色不是空白的方块
{
fourBlocks[0] = blocks[clicked_i-search][clicked_j]; // 赋给这个存储的数组
break;
}
}
// 向下找
for (search=0;clicked_i+search<RowNum;search++)
{
if (blocks[clicked_i+search][clicked_j].colorId!=0) // 找到第一个颜色不是空白的方块
{
fourBlocks[1] = blocks[clicked_i+search][clicked_j]; // 赋给这个存储的数组
break;
}
}
// 向左找
for (search=0;clicked_j-search>=0;search++)
{
if (blocks[clicked_i][clicked_j-search].colorId!=0) // 找到第一个颜色不是空白的方块
{
fourBlocks[2] = blocks[clicked_i][clicked_j-search]; // 赋给这个存储的数组
break;
}
}
// 向右找
for (search=0;clicked_j+search<ColNum;search++)
{
if (blocks[clicked_i][clicked_j+search].colorId!=0) // 找到第一个颜色不是空白的方块
{
fourBlocks[3] = blocks[clicked_i][clicked_j+search]; // 赋给这个存储的数组
break;
}
}
// 统计fourBlocks的四个小方块,有没有同样颜色数目大于等于2的
int colorStatistics[ColorTypeNum+1] = {0}; // 初始化个数为0
int isBadClick = 1; // 假设点击的方块不合适,十字区域没有有效消除的方块
for (i=1;i<ColorTypeNum+1;i++) // i=0是空白颜色,不要统计
{
for (j=0;j<4;j++) // 遍历fourBlocks
{
if (fourBlocks[j].colorId==i)
colorStatistics[i]++; // 方块颜色为非零的i的话,把对应的统计个数+1
}
if (colorStatistics[i]>=2) // 如果这种颜色方块个数大于等于2
{
isBadClick = 0; // 能消除了,这次点击是好的操作
// 把对应十字区域要消除的方块颜色改成空白颜色
for (j=0;j<4;j++) // 遍历fourBlocks
{
if (fourBlocks[j].colorId==i)
{
// 要消除的方块区域绘制提示框
drawBlockHint(fourBlocks[j].i,fourBlocks[j].j,RGB(0,0,0),0);
// 颜色序号设为0,也就是空白的灰白色
blocks[fourBlocks[j].i][fourBlocks[j].j].colorId = 0;
}
}
score += colorStatistics[i]; // 得分加上消除的方块数
}
}
// 点击的方块,十字区域没有能消除的方块,为错误点击,减去10秒钟时间
if (isBadClick==1)
totalTime -= 10;
FlushBatchDraw(); // 批量绘制
Sleep(300); // 绘制好提示框后暂停300毫秒
} // while 当按下鼠标左键时
}
}
int main() // 主函数运行
{
startup();
while (1)
{
show();
updateWithoutInput();
updateWithInput();
}
closegraph();
return 0;
}
这一节主要讲解了指针的相关语法知识,学习了倒计时的方法,实现了十字消除游戏。读者可以尝试在本章代码基础上继续改进:
1、实现随着游戏的进行,通过关卡要求消除方块的比例越来越高;
2、利用文件读写,实现关卡数据与最高分的记录与读取。
读者也可以参考本章的开发思路,尝试设计并分步骤实现消消乐、消灭星星、宝石迷阵等各种消除类游戏。
作者:童晶
希望对大家有帮助!
此外,我也给大家分享我收集的其他资源,从最零基础开始的教程到C语言C++项目案例,帮助大家在学习C语言的道路上披荆斩棘!
编程学习书籍分享:
编程学习视频分享:
整理分享(多年学习的源码、项目实战视频、项目笔记,基础入门教程)最重要的是你可以在群里面交流提问编程问题哦!
对于C/C++感兴趣可以关注小编在后台私信我:【编程交流】一起来学习哦!可以领取一些C/C++的项目学习视频资料哦!已经设置好了关键词自动回复,自动领取就好了!
相关推荐
- go语言也可以做gui,go-fltk让你做出c++级别的桌面应用
-
大家都知道go语言生态并没有什么好的gui开发框架,“能用”的一个手就能数的清,好用的就更是少之又少。今天为大家推荐一个go的gui库go-fltk。它是通过cgo调用了c++的fltk库,性能非常高...
- 旧电脑的首选系统:TinyCore!体积小+精简+速度极快,你敢安装吗
-
这几天老毛桃整理了几个微型Linux发行版,准备分享给大家。要知道可供我们日常使用的Linux发行版有很多,但其中的一些发行版经常会被大家忽视。其实这些微型Linux发行版是一种非常强大的创新:在一台...
- codeblocks和VS2019下的fltk使用中文
-
在fltk中用中文有点问题。英文是这样。中文就成这个样子了。我查了查资料,说用UTF-8编码就行了。edit->Fileencoding->UTF-8然后保存文件。看下下边的编码指示确...
- FLTK(Fast Light Toolkit)一个轻量级的跨平台Python GUI库
-
FLTK(FastLightToolkit)是一个轻量级的跨平台GUI库,特别适用于开发需要快速、高效且简单界面的应用程序。本文将介绍Python中的FLTK库,包括其特性、应用场景以及如何通过代...
- 中科院开源 RISC-V 处理器“香山”流片,已成功运行 Linux
-
IT之家1月29日消息,去年6月份,中科院大学教授、中科院计算所研究员包云岗,发布了开源高性能RISC-V处理器核心——香山。近日,包云岗在社交平台晒出图片,香山芯片已流片,回片后...
- Linux 5.13内核有望合并对苹果M1处理器支持的初步代码
-
预计Linux5.13将初步支持苹果SiliconM1处理器,不过完整的支持工作可能还需要几年时间才能完全完成。虽然Linux已经可以在苹果SiliconM1上运行,但这需要通过一系列的补丁才能...
- Ubuntu系统下COM口测试教程(ubuntu port)
-
1、在待测试的板上下载minicom,下载minicom有两种方法:方法一:在Ubuntu软件中心里面搜索下载方法二:按“Ctrl+Alt+T”打开终端,打开终端后输入“sudosu”回车;在下...
- 湖北嵌入式软件工程师培训怎么选,让自己脱颖而出
-
很多年轻人毕业即失业、面试总是不如意、薪酬不满意、在家躺平。“就业难”该如何应对,参加培训是否能改变自己的职业走向,在湖北,有哪些嵌入式软件工程师培训怎么选值得推荐?粤嵌科技在嵌入式培训领域有十几年经...
- 新阁上位机开发---10年工程师的Modbus总结
-
前言我算了一下,今年是我跟Modbus相识的第10年,从最开始的简单应用到协议了解,从协议开发到协议讲解,这个陪伴了10年的协议,它一直没变,变的只是我对它的理解和认识。我一直认为Modbus协议的存...
- 创建你的第一个可运行的嵌入式Linux系统-5
-
@ZHangZMo在MicrochipBuildroot中配置QT5选择Graphic配置文件增加QT5的配置修改根文件系统支持QT5修改output/target/etc/profile配置文件...
- 如何在Linux下给zigbee CC2530实现上位机
-
0、前言网友提问如下:粉丝提问项目框架汇总下这个网友的问题,其实就是实现一个网关程序,内容分为几块:下位机,通过串口与上位机相连;下位机要能够接收上位机下发的命令,并解析这些命令;下位机能够根据这些命...
- Python实现串口助手 - 03串口功能实现
-
串口调试助手是最核心的当然是串口数据收发与显示的功能,pzh-py-com借助的是pySerial库实现串口收发功能,今天痞子衡为大家介绍pySerial是如何在pzh-py-com发挥功能的。一、...
- 为什么选择UART(串口)作为调试接口,而不是I2C、SPI等其他接口
-
UART(通用异步收发传输器)通常被选作调试接口有以下几个原因:简单性:协议简单:UART的协议非常简单,只需设置波特率、数据位、停止位和校验位就可以进行通信。相比之下,I2C和SPI需要处理更多的通...
- 同一个类,不同代码,Qt 串口类QSerialPort 与各种外设通讯处理
-
串口通讯在各种外设通讯中是常见接口,因为各种嵌入式CPU中串口标配,工业控制中如果不够还通过各种串口芯片进行扩展。比如spi接口的W25Q128FV.对于软件而言,因为驱动接口固定,软件也相对好写,因...
- 嵌入式linux为什么可以通过PC上的串口去执行命令?
-
1、uboot(负责初始化基本硬bai件,如串口,网卡,usb口等,然du后引导系统zhi运行)2、linux系统(真正的操作系统)3、你的应用程序(基于操作系统的软件应用)当你开发板上电时,u...
你 发表评论:
欢迎- 一周热门
- 最近发表
-
- go语言也可以做gui,go-fltk让你做出c++级别的桌面应用
- 旧电脑的首选系统:TinyCore!体积小+精简+速度极快,你敢安装吗
- codeblocks和VS2019下的fltk使用中文
- FLTK(Fast Light Toolkit)一个轻量级的跨平台Python GUI库
- 中科院开源 RISC-V 处理器“香山”流片,已成功运行 Linux
- Linux 5.13内核有望合并对苹果M1处理器支持的初步代码
- Ubuntu系统下COM口测试教程(ubuntu port)
- 湖北嵌入式软件工程师培训怎么选,让自己脱颖而出
- 新阁上位机开发---10年工程师的Modbus总结
- 创建你的第一个可运行的嵌入式Linux系统-5
- 标签列表
-
- wireshark怎么抓包 (75)
- qt sleep (64)
- cs1.6指令代码大全 (55)
- factory-method (60)
- sqlite3_bind_blob (52)
- hibernate update (63)
- c++ base64 (70)
- nc 命令 (52)
- wm_close (51)
- epollin (51)
- sqlca.sqlcode (57)
- lua ipairs (60)
- tv_usec (64)
- 命令行进入文件夹 (53)
- postgresql array (57)
- statfs函数 (57)
- .project文件 (54)
- lua require (56)
- for_each (67)
- c#工厂模式 (57)
- wxsqlite3 (66)
- dmesg -c (58)
- fopen参数 (53)
- tar -zxvf -c (55)
- 速递查询 (52)