C语言——2048完整版

2048是一个简单又有趣的小游戏,相信大家都接触并了解过,那如何通过代码来实现他呢?下面就让我们来一起看看。

目录

1、头文件

2、主函数

3、 StarGame

 4、GetNum

 5、Show

6、Picture

 7、GetButton

8、MergeLeft

 9、MergeUp

10、MergeRight

11、MergeDown

 12、MergeNum

 13、IsGameover

 14、Run

15、最后附上运行结果一张


1、头文件

其中<graphics.h>是为了使用easyx图形工具,使我们的游戏最终成品得以图像化,<time.h>是为了生成随机种子,保证每一次所出现的数字足够具有随机性。同时还需要两个宏定义,来规范二维数组。以及各个方向的键盘扫描码。

#define _CRT_SECURE_NO_WARNINGS //这一句必须放在第一行
#include <stdio.h>
#include <assert.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <conio.h>
#include <graphics.h>		// 引用图形库头文件

#define ROW 4
#define COL ROW
#define KEY_UP 72                                                                 /*  方向键'上'的扫描码码值     */
#define KEY_DOWN 80                                                          /*  方向键'下'的扫描码码值     */
#define KEY_LEFT  75                                                             /*  方向键'左'的扫描码码值     */
#define KEY_RIGHT 77                                                           /*  方向键'右'的扫描码码值     */

2、主函数

首先我们先建立一个图形窗口,设置为450*450像素,并且在代码最后关闭绘图窗口,然后设置一个数组,用来存放2048中的每一个数字。一切准备工作做好之后我们就可以调用StarGame与Run函数了。下面我们来一步一步实现这两个函数

int main()
{
	initgraph(450, 450);	// 创建绘图窗口,大小为 450x450 像素

	int arr [ROW][COL] = { 0 };

	StarGame(arr);
	
	Run(arr);//运行游戏

	closegraph();			// 关闭绘图窗口
}

3、 StarGame

想一想2048,游戏开始时会在随机位置生成一个数字,并且打印在屏幕上,所以在这个函数中我们需要“获取随机数字”GetNum,与“打印”Show。

void StarGame(int (*arr)[COL])
{
	GetNum(arr);
	GetNum(arr);//提供一个随机数字
	Show(arr);
}

 4、GetNum

在这个函数中我们通过两步随机数字组合套路对坐标随机性进行控制,再将生成数字的可能性进行调整,最后进行线性检查,如果生成的随机坐标的位置上已经有数字了就顺延到下一位,如果已经到了末尾就顺延到下一行

void GetNum(int(*arr)[COL])
{
	static int seed=0;//静态变量不会销毁每次都是一样的值
	srand((unsigned int)time(NULL) + seed);//上面两步是随机数字套路,产生随机种子
	seed++;
	//坐标随机
	int row = rand() % ROW;
	int col = rand() % COL;
	//数字随机
	int num = 2;
	if (rand() % 5 == 0)//把4的概率置为1/5
		num = 4;
	while (arr[row][col]!=0)
	{
		col++;
		if (col == COL)
		{
			row =(row + 1) % ROW;
			col = 0;
		}
	}//线性检查,如果第row行col列有数字就顺延到下一个
	arr[row][col] = num;
}

 5、Show

加载背景图片,并且打印在屏幕上,注意这里数字也需要背景图片,所以引入一个新函数Picture。

void Show(int(*arr)[COL])//打印
{
	//system("cls");
	IMAGE img;
	loadimage(&img, _T("bk.png"));
	putimage(0, 0, &img);
	for (int i = 0; i < ROW; i++)
	{
		for (int j = 0; j < COL; j++)
		{
			Picture(arr, i, j);//输出数字对应的照片
		}
	}
}

6、Picture

如果数字为“0”时则不需要调用图片,如果数字为“2”等其他数字时则需要调用图片,注意这里的loadimage为easyx的函数。如果有需要这些数字图片的,可以私信我。

bool Picture(int(*arr)[COL], int row, int col)
{
	IMAGE img;
	switch (arr[row][col])
	{
	case 0://不需要加载图片
		return false;
	case 2:
		loadimage(&img, _T("2.jpg"), 100, 100, true); //这些图片的大小都超过100,100,需要进行相应的缩小
		break;
	case 4:
		loadimage(&img, _T("4.jpg"), 100, 100, true);
		break;
	case 8:
		loadimage(&img, _T("8.jpg"), 100, 100, true);
		break;
	case 16:
		loadimage(&img, _T("16.jpg"), 100, 100, true);
		break;
	case 32:
		loadimage(&img, _T("32.jpg"), 100, 100, true);
		break;
	case 64:
		loadimage(&img, _T("64.jpg"), 100, 100, true);
		break;
	case 128:
		loadimage(&img, _T("128.jpg"), 100, 100, true);
		break;
	case 256:
		loadimage(&img, _T("256.jpg"), 100, 100, true);
		break;
	case 512:
		loadimage(&img, _T("512.jpg"), 100, 100, true);
		break;
	case 1024:
		loadimage(&img, _T("1024.jpg"), 100, 100, true);
		break;
	case 2048:
		loadimage(&img, _T("2048.jpg"), 100, 100, true);
		break;
	default:
		break;
	}

	putimage(10 + col * 110, 10 + row * 110, &img);
	return true;
}

 7、GetButton

想要控制游戏的方向,就需要从键盘读取,所以在这个函数里我们来进行“方向获取”这项工作。

int GetButton()
{
	ExMessage m;
	while (peekmessage(&m, EX_KEY))
	{
		if (m.message == WM_KEYDOWN)
		{
			switch (m.vkcode)
			{
			case VK_LEFT:
				return 1;
			case VK_UP:
				return 2;
			case VK_RIGHT:
				return 3;
			case VK_DOWN:
				return 4;
			}
		}
	}
	return 0;
}

接下来我们来上下左右四个方向来添加代码

8、MergeLeft

bool MergeLeft(int(*arr)[COL])
{
	int index;
	bool flg = false;
	for (int i = 0; i < ROW; i++)
	{
		index = -1;
		for (int j = 0; j < COL; j++)
		{
			if (arr[i][j] != 0 && index == -1)
				index = j;//如果第ij个数字不是0,并且第一个值还没找到,所以把index赋值成当前数字下标
			else if (arr[i][j] != 0 && arr[i][j] != arr[i][index])
				index = j;//如果第ij个数字不是0,并且与找到的前一个数字不相等,则抛弃上一个
			else if (arr[i][j] != 0)
			{
				arr[i][index] *= 2;
				arr[i][j] = 0;//如果第ij个数字不是0,并且与找到的值相等,给找到的值*2,并把当前值赋为0
				flg = true;//如果发生了合并
			}
		}


		 index = 0;
		//可以保存数据的项的下标
		for (int j = 0; j < COL; j++)
		{
				if (arr[i][j] != 0)//需要移动
				{
					arr[i][index] = arr[i][j];
					if (index != j)
					{
						arr[i][j] = 0;
						flg = true;//如果发生了移动
					}
						 index++;
				}	
		}
	}
	return flg;
}

 9、MergeUp

bool MergeUp(int(*arr)[COL])
{
	int index;//第一个非0值的下标(行)
	bool flg = false;
	for (int j = 0; j < COL; j++)//列
	{
		index = -1;
		for (int i = 0; i < ROW; i++)//行,合并数据
		{
			if (arr[i][j] != 0 && index == -1)
				index = i;
			else if (arr[i][j] != 0 && arr[i][j] != arr[index][j])
				index = i;
			else if (arr[i][j] != 0)//合并
			{
				arr[index][j] *= 2;
				arr[i][j] = 0;
				flg = true;
			}
		}
		index = 0;//可以存放数据的下标(行)
		for (int i = 0; i < ROW; i++)//移动数据
		{
			if (arr[i][j] != 0)
			{
				arr[index][j] = arr[i][j];
				if (index != i)
				{
					arr[i][j] = 0;
					flg = true;
				}
				index++;
			}
		}
	}
	return flg;
}

10、MergeRight

bool MergeRight(int(*arr)[COL])
{
	int index;
	bool flg = false;
	for (int i = 0; i <ROW; i++)
	{
		index = -1;
		for (int j = 3; j >=0; j--)
		{
			if (arr[i][j] != 0 && index == -1)
				index = j;//如果第ij个数字不是0,并且第一个值还没找到,所以把index赋值成当前数字下标
			else if (arr[i][j] != 0 && arr[i][j] != arr[i][index])
				index = j;//如果第ij个数字不是0,并且与找到的前一个数字不相等,则抛弃上一个
			else if (arr[i][j] != 0)
			{
				arr[i][index] *= 2;
				arr[i][j] = 0;//如果第ij个数字不是0,并且与找到的值相等,给找到的值*2,并把当前值赋为0
				flg = true;//如果发生了合并
			}
		}


		index = 3;
		//可以保存数据的项的下标
		for (int j =3; j >=0; j--)
		{
			if (arr[i][j] != 0)//需要移动
			{
				arr[i][index] = arr[i][j];
				if (index != j)
				{
					arr[i][j] = 0;
					flg = true;//如果发生了移动
				}
				index--;
			}
		}
	}
	return flg;
}

11、MergeDown

bool MergeDown(int(*arr)[COL])
{
	int index;//第一个非0值的下标(行)
	bool flg = false;
	for (int j = 0; j<COL; j++)//列
	{
		index = -1;
		for (int i =3; i >=0; i--)//行,合并数据
		{
			if (arr[i][j] != 0 && index == -1)
				index = i;
			else if (arr[i][j] != 0 && arr[i][j] != arr[index][j])
				index = i;
			else if (arr[i][j] != 0)//合并
			{
				arr[index][j] *= 2;
				arr[i][j] = 0;
				flg = true;
			}
		}

		index = 3;//可以存放数据的下标(行)
		for (int i = 3; i>=0; i--)//移动数据
		{
			if (arr[i][j] != 0)
			{
				arr[index][j] = arr[i][j];
				if (index != i)
				{
					arr[i][j] = 0;
					flg = true;
				}
				index--;
			}
		}
	}
	return flg;
}

 12、MergeNum

保存方向

bool MergeNum(int(*arr)[COL], int direct)
{
	bool flg = false;
	switch (direct)//当()里的表达式和某一个case值相同时进入
	{
	case 1://向左
		flg = MergeLeft(arr);
		break;
	case 2://向上
		flg= MergeUp(arr);
		break;
	case 3://向右
		flg = MergeRight(arr);
		break;
	case 4://向下
		flg = MergeDown(arr);
		break;
	default:
		break;
	}
	return flg;
}

 13、IsGameover

判断游戏是否结束

bool IsGameover(int(*arr)[COL])
{
	int count0 = 0;//统计0的个数
	int i;
	int j;
	for (i = 0; i < ROW; i++)//判断是否有空格
	{
		for (j = 0; j < COL; j++)
		{
			if (arr[i][j] == 0)
				count0++;
		}
	}
	if (count0 != 0)//还有空格
		return false;//游戏没有结束
	for (i = 0; i < ROW; i++)//是否有相邻的数字一样
	{
		for (j = 0; j < COL; j++)
		{
			if (j + 1 < COL)//判断当前的值和右边是否相同
				if (arr[i][j] == arr[i][j + 1])
					return false;//游戏没有结束
			if (i + 1 < ROW)//判断当前的值和下边是否相同
				if (arr[i][j] == arr[i + 1][j])
					return false;//游戏没有结束
		}
	}
	return true;
}

 14、Run

最后就是我们刚开始提到的Run函数

void Run(int(*arr)[COL])
{
	int direct;//从键盘获得一个方向
	while (1)
	{
		direct= GetButton();
		//if (!MergeNum(arr, direct))
		//	continue;//如果没有发生合并或移动就不产生新的数据
		if (!MergeNum(arr, direct))
			continue;
		GetNum(arr);
		Show(arr);
		if (IsGameover(arr))
		{
			return;//判断游戏是否结束
		}
	}
}

15、最后附上运行结果一张

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/228741.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

python圣诞树代码编程

以下是一个简单的Python圣诞树代码&#xff1a; def draw_tree(height): for i in range(height): print( * (height - i - 1) * * (2 * i 1)) print( * (height - 1) |)draw_tree(10) 这个函数会绘制一个等腰三角形&#xff0c;其中每一行的星号数量从1开…

改造后的LED电子大屏幕:环保与节能的新选择

随着人们生活水平和文明程度的提高&#xff0c;对于环保的意识也日益增强。这在支付宝几亿用户参与蚂蚁森林环保项目中体现得淋漓尽致&#xff0c;为环保事业贡献一份力量。为迎合这一时代发展的要求&#xff0c;LED显示屏厂家早已推出了亮度可调的LED节能屏。如今&#xff0c;…

linux高级篇基础理论七(Tomcat)

♥️作者&#xff1a;小刘在C站 ♥️个人主页&#xff1a; 小刘主页 ♥️不能因为人生的道路坎坷,就使自己的身躯变得弯曲;不能因为生活的历程漫长,就使求索的 脚步迟缓。 ♥️学习两年总结出的运维经验&#xff0c;以及思科模拟器全套网络实验教程。专栏&#xff1a;云计算技…

html、css类名命名思路整理

开发页面时&#xff0c;老是遇到起名问题&#xff0c;越想越头疼&#xff0c;严重影响开发进度&#xff0c;都是在想名字&#xff0c;现在做一下梳理&#xff0c;统一一下思想&#xff0c;希望以后能减少这块的痛苦。 命名规则 [功能名称]__[组成部分名称]--[样式名称] 思路…

Spring Security 6.x 系列(10)—— SecurityConfigurer 配置器及其分支实现源码分析(二)

一、前言 在本系列文章&#xff1a; Spring Security 6.x 系列&#xff08;4&#xff09;—— 基于过滤器链的源码分析&#xff08;一&#xff09; 中着重分析了Spring Security在Spring Boot自动配置、 DefaultSecurityFilterChain和FilterChainProxy 的构造过程。 Spring …

神经内科临床常用的焦虑、抑郁评估量表,医生必备!

根据神经内科医生的量表使用情况&#xff0c;常笑医学整理了神经内科临床上常用的焦虑、抑郁评估量表&#xff0c;为大家分享临床常见的焦虑、抑郁、睡眠等量表评估内容&#xff0c;均支持量表下载和在线使用&#xff0c;建议收藏&#xff01; 1.汉密顿抑郁量表&#xff08;Ham…

从零开发短视频电商 AWS OpenSearch Service开发环境申请以及Java客户端介绍

文章目录 创建域1.创建域2.输入配置部署选项数据节点网络精细访问控制访问策略 获取域端点数据如何插入到OpenSearch ServiceJava连接OpenSearch Servicespring-data-opensearchelasticsearch-rest-high-level-clientopensearch-rest-clientopensearch-java 因为是开发测试使用…

喜讯!云起无垠上榜《成长型初创企业推荐10强》

近期&#xff0c;由中国计算机学会抗恶劣环境计算机专业委员会、信息产业信息安全测评中心和安全牛联合发起的第十一版《中国网络安全企业100强》榜单正式发布。在这份备受关注的榜单中&#xff0c;云起无垠凭借其创新的技术能力&#xff0c;荣登《成长型初创企业推荐10强》榜单…

echarts绘制一个环形图

其他echarts&#xff1a; echarts绘制一个柱状图&#xff0c;柱状折线图 echarts绘制一个饼图 echarts绘制一个环形图2 效果图&#xff1a; 代码&#xff1a; <template><div class"wrapper"><!-- 环形图 --><div ref"doughnutChart…

CCKS2023-面向金融领域的主体事件检测-亚军方案分享

赛题分析 大赛地址 https://tianchi.aliyun.com/competition/entrance/532098/introduction?spma2c22.12281925.0.0.52b97137bpVnmh 任务描述 主体事件检测是语言文本分析和金融领域智能应用的重要任务之一&#xff0c;如在金融风控领域往往会对公司主体进行风险事件的检测…

【C#设计模式 + Filter】装饰器模式专项——过滤器

c# 装饰器模式专项——过滤器 装饰器模式专项——过滤器Filter1.winform实现通过特性改控件名称&#xff08;.Framework)2.手写过滤器 (.NET Core) 装饰器模式专项——过滤器Filter 左边为api启动流程。 右边为需要实现的winform启动流程。右边大框里面需要我们手动实现。 1.wi…

python pyaudio实时读取音频数据并展示波形图

python pyaudio实时读取音频数据并展示波形图 下面代码可以驱动电脑接受声音数据&#xff0c;并实时展示音波图&#xff1a; import numpy as np import matplotlib.pyplot as plt import matplotlib.animation as animation import pyaudio import wave import os import op…

Michael.W基于Foundry精读Openzeppelin第41期——ERC20Capped.sol

Michael.W基于Foundry精读Openzeppelin第41期——ERC20Capped.sol 0. 版本0.1 ERC20Capped.sol 1. 目标合约2. 代码精读2.1 constructor() && cap()2.2 _mint(address account, uint256 amount) 0. 版本 [openzeppelin]&#xff1a;v4.8.3&#xff0c;[forge-std]&…

2023年全球软件开发大会(QCon广州站2023)-核心PPT资料下载

一、峰会简介 本次峰会包含&#xff1a;泛娱乐时代的边缘计算与通讯、稳定性即生命线、下一代软件架构、出海的思考、现代数据架构、AGI 与 AIGC 落地、大前端技术探索、编程语言实战、DevOps vs 平台工程、新型数据库、AIGC 浪潮下的企业出海、AIGC 浪潮下的效能智能化、数据…

使用Rust 构建C 组件

协议解析&#xff0c;这不就很快了&#xff0c;而且原生的标准库红黑树和avl 树支持&#xff0c;异步tokio 这些库&#xff0c;编写应用组件就很快了 rust 标准库不支持 unix 的消息队列&#xff0c;但是支持 shm 和 uds&#xff0c;后者从多方面考虑都比&#xff0c;消息队列更…

python爬取 HTTP_2 网站超时问题的解决方案

问题背景 在进行网络数据爬取时&#xff0c;使用 Python 程序访问支持 HTTP/2 协议的网站时&#xff0c;有时会遇到超时问题。这可能会导致数据获取不完整&#xff0c;影响爬虫程序的正常运行。 问题描述 在实际操作中&#xff0c;当使用 Python 编写的爬虫程序访问支持 HTT…

SAP物料会计视图的价格确定MLAST为空后台补录

物料10013812 工厂会计视图的价格确定为空&#xff0c;前台目前无法修改&#xff0c;申请修改底表&#xff0c;将价格确定调整为2 解决&#xff1a; 该字段涉及&#xff1a;MBEW表和CKMLHD表的MLAST字段两个地方&#xff0c;经修改后&#xff0c;前后台数据一致。 只改技术信…

2023_Spark_实验二十五:SparkStreaming读取Kafka数据源:使用Direct方式

SparkStreaming读取Kafka数据源&#xff1a;使用Direct方式 一、前提工作 安装了zookeeper 安装了Kafka 实验环境&#xff1a;kafka zookeeper spark 实验流程 二、实验内容 实验要求&#xff1a;实现的从kafka读取实现wordcount程序 启动zookeeper zk.sh start# zk.sh…

性能测试(超详细)

近期公司为了节省成本搞了一波机房迁移&#xff0c;整合了一些南美部署架构。有一些上google云和有些下阿里云等大的调整。 在做机房迁移项目当中就需要思考如何进行性能测试&#xff0c;这种大的机房迁移SRE&#xff08;运维&#xff09;会针对组件会做一些单组件的性能测试&a…

服务器配置免密SSH

在当今互联网时代&#xff0c;远程工作和网络安全已成为信息技术领域的热点话题。无论是管理远程服务器、维护网络设备还是简单地从家中连接到办公室&#xff0c;安全始终是首要考虑的因素。这就是为什么 SSH&#xff08;Secure Shell&#xff09;成为了网络专业人士的首选工具…