贪吃蛇游戏(代码篇)

我们并不是为了满足别人的期待而活着。

前言 

  这是我自己做的第五个小项目---贪吃蛇游戏(代码篇)。后期我会继续制作其他小项目并开源至博客上。 

  上一小项目是贪吃蛇游戏(必备知识篇),没看过的同学可以去看看:

有关贪吃蛇必备知识的小项目icon-default.png?t=O83Ahttps://blog.csdn.net/hsy1603914691/article/details/142455297?sharetype=blogdetail&sharerId=142455297&sharerefer=PC&sharesource=hsy1603914691&spm=1011.2480.3001.8118

实现代码

1. 下面代码直接复制即可运行。

2. 每个代码块都有简洁的总结和解释。

<snake.h>文件

#define _CRT_SECURE_NO_WARNINGS
#include <locale.h>
#include <stdio.h>
#include <windows.h>
#include <stdbool.h>
#include <stdlib.h>
#include <time.h>

#define KEY_PRESS(VK) ((GetAsyncKeyState(VK)&0x1) ? 1 : 0) //设置键值
#define POS_X 24 //蛇初始位置
#define POS_Y 5 //蛇初始位置

//节点类型
typedef struct SnakeNode
{
	//节点的坐标
	int x;
	int y;
	//指向下一个节点的指针
	struct SnakeNode* next;
}SnakeNode;
typedef struct SnakeNode*  pSnakeNode;

//贪吃蛇的信息
typedef struct Snake
{
	pSnakeNode _pSnake;//贪吃蛇的身体节点
	pSnakeNode _pFood;//食物节点
	enum Direction _dir;//贪吃蛇的方向
	enum Game_Statues _status;//贪吃蛇的状态
	int _food_weight;//一个食物的分数
	int _score;//总分数
	int _sleep_time;//休息时间,即贪吃蛇的速度
}Snake;
typedef struct Snake* pSnake;

//方向
enum Direction
{
	UP,
	DOWN,
	LEFT,
	RIGHT
};

//状态
enum Game_Status
{
	OK,
	KILL_BY_WALL,
	KILL_BY_SELF,
	END_NORMAL
};

//游戏开始
void GameStart(pSnake ps);
//欢迎函数
void WelcomeToGame();
//定位坐标
void SetPos(int x, int y);
//打印地图
void CreateMap();
//初始化贪吃蛇
void InitSnake(pSnake ps);
//创造食物
void CreateFood(pSnake ps);
//游戏运行
void GameRun(pSnake ps);
//打印帮助信息
void PrintHelpInfo();
//暂停设置
void Pause();
//实现贪吃蛇的移动
void SnakeMove(pSnake ps);
//判断是否吃到食物
int NextIsFood(pSnakeNode pn, pSnake ps);
//实现贪吃蛇吃食物并增长蛇身
void EatFood(pSnakeNode pn, pSnake ps);
//吃到食物后使食物消失
void NoFood(pSnakeNode pn, pSnake ps);
//被墙杀死
void KillByWall(pSnake ps);
//被自己杀死
void KillBySelf(pSnake ps);
//游戏结束
void GameEnd(pSnake ps);

<snake.c>文件

#include "snake.h"
//游戏开始
void GameStart(pSnake ps)
{
	//设置窗口
	system("mode con cols=100 lines=30");//调整CMD行与列
	system("title 贪吃蛇");//修改CMD的标题

	//获取标准输出的句柄,存放在houtput中。
	HANDLE houtput = GetStdHandle(STD_OUTPUT_HANDLE);
	if (houtput == INVALID_HANDLE_VALUE) // 处理错误,例如输出错误信息
	{
		fprintf(stderr, "Failed to get standard output handle.\n");
		return;
	}

	//创建一个CONSOLE_CURSOR_INFO的结构体
	CONSOLE_CURSOR_INFO CursorInfo;
	if (!GetConsoleCursorInfo(houtput, &CursorInfo)) // 处理错误,例如输出错误信息
	{
		fprintf(stderr, "Failed to get console cursor info.\n");
		return;
	}

	//隐藏控制台光标
	CursorInfo.bVisible = false; 
	if (!SetConsoleCursorInfo(houtput, &CursorInfo)) // 处理错误,例如输出错误信息
	{
		fprintf(stderr, "Failed to set console cursor info.\n");
		return;
	}
	
	//欢迎函数
	WelcomeToGame();
	//打印地图
	CreateMap();
	//初始化贪吃蛇
	InitSnake(ps);
	//设置食物的位置
	CreateFood(ps);
}

//欢迎函数
void WelcomeToGame()
{
	SetPos(32, 13);
	printf("Welcome to the Classic Snake Game!");
	SetPos(39, 22);
	system("pause");//打印完一个界面后直接暂停,直到点击继续
	system("cls");//在清空界面,打印新的一个界面
	SetPos(30, 13);
	wprintf(L"Navigate the Snake using ↑ ↓ ← →.");
	SetPos(33, 15);
	wprintf(L"Accelerate to earn more points.");
	SetPos(38, 23);
	system("pause");
	system("cls");
}

//定位坐标
void SetPos(int x, int y)
{
	//获取标准输出的句柄,存放在houtput中
	HANDLE houtput = GetStdHandle(STD_OUTPUT_HANDLE);
	//设定我们想要定位的坐标
	COORD pos = { x,y };
	//将光标定位到pos2
	SetConsoleCursorPosition(houtput, pos);
}

//打印地图
void CreateMap()
{
	int i = 0;
	//打印上边界
	for (i = 0; i < 29; i++)
	{
		wprintf(L"□");
	}
	//打印下边界
	SetPos(0, 26);
	for (i=0; i < 29; i++)
	{
		wprintf(L"□");
	}
	//打印左边界
	for (i = 1; i <= 25; i++)
	{
		SetPos(0, i);
		wprintf(L"□");
	}
	//打印右边界
	for (i = 1; i <= 25; i++)
	{
		SetPos(56, i);
		wprintf(L"□");
	}
}

//初始化贪吃蛇
void InitSnake(pSnake ps)
{
	int i = 0;
	for (i = 0; i < 5; i++)//开始贪吃蛇一共设置五个长度
	{
		pSnakeNode  cur = (pSnakeNode)malloc(sizeof(SnakeNode));
		if (cur == NULL)
		{
			perror("InitSnake error");
			exit(1);
		}
		cur->next = NULL;
		cur->x = POS_X + 2 * i;
		cur->y = POS_Y;
		//头插法
		if (ps->_pSnake == NULL)
		{
			ps->_pSnake = cur;
		}
		else
		{
			cur->next = ps->_pSnake;
			ps->_pSnake = cur;
		}
	}
	pSnakeNode  cur = ps->_pSnake;
	while (cur != NULL)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"●");
		cur = cur->next;
	}
	//设置贪吃蛇的属性
	ps->_dir = RIGHT;
	ps->_score = 0;
	ps->_food_weight = 10;
	ps->_sleep_time = 200;
	ps->_status = OK;
}

//设置食物的位置
void CreateFood(pSnake ps)
{
	int x;
	int y;
again:
	do
	{
		x = (rand()) % 53 + 2;
		y = (rand()) % 25 + 1;
	} while (x % 2 != 0);
	//不能与蛇身冲突
	pSnakeNode  cur = ps->_pSnake;
	while (cur != NULL)
	{
		if ((x == cur->x) && (y == cur->y))
		{
			goto again;
		}
		cur = cur->next;
	}
	//创建食物节点
	pSnakeNode  pFood = (pSnakeNode)malloc(sizeof(SnakeNode));
	if (pFood == NULL)
	{
		perror("CreateFood error");
		exit(1);
	}
	pFood->x = x;
	pFood->y = y;
	pFood->next = NULL;
	SetPos(x, y);
	wprintf(L"★");
	ps->_pFood = pFood;
}

//游戏运行
void GameRun(pSnake ps)
{
	//打印欢迎界面
	PrintHelpInfo();
	//游戏开始运行
	do
	{
		//显示分数
		SetPos(64, 7);
		printf("Current score: %d", ps->_score);
		SetPos(64, 8);
		printf("Current food score: %2d", ps->_food_weight);
		//判断玩家操作
		if (KEY_PRESS(VK_UP) && (ps->_dir != UP))
		{
			ps->_dir = UP;
		}
		else if (KEY_PRESS(VK_DOWN) && (ps->_dir != DOWN))
		{
			ps->_dir = DOWN;
		}
		else if (KEY_PRESS(VK_LEFT) && (ps->_dir != LEFT))
		{
			ps->_dir = LEFT;
		}
		else if (KEY_PRESS(VK_RIGHT) && (ps->_dir != RIGHT))
		{
			ps->_dir = RIGHT;
		}
		else if (KEY_PRESS(VK_SPACE))
		{
			Pause();//暂停设置
		}
		else if (KEY_PRESS(VK_ESCAPE))
		{
			ps->_status = END_NORMAL;
		}
		else if (KEY_PRESS(VK_F3))
		{
			if (ps->_sleep_time > 80)
			{
				ps->_sleep_time -= 30;
				ps->_food_weight += 2;
			}
		}
		else if (KEY_PRESS(VK_F4))
		{
			if (ps->_food_weight > 2)
			{
				ps->_sleep_time += 30;
				ps->_food_weight -= 2;
			}
		}
		//实现贪吃蛇的移动
		SnakeMove(ps);
		//通过短暂暂停来展现动态效果
		Sleep(ps->_sleep_time);
	} while (ps->_status == OK);
}

//打印欢迎界面
void PrintHelpInfo()
{
	SetPos(64, 10);
	wprintf(L"No wall passing. No self-biting.");
	SetPos(64, 12);
	wprintf(L"F3 to speed up. F4 to slow down.");
	SetPos(64, 14);
	wprintf(L"ESC to exit. Space to pause.");
	SetPos(74, 21);
	wprintf(L"Made by HSY,");
	SetPos(66, 22);
	wprintf(L"a uniquely independent pig.");
}

//暂停设置
void Pause()
{
	while (1)
	{
		Sleep(200);
		if (KEY_PRESS(VK_SPACE))
		{
			break;
		}
	}
}

//实现贪吃蛇的移动
void SnakeMove(pSnake ps)
{
	pSnakeNode  pNextNode = (pSnakeNode)malloc(sizeof(SnakeNode));
	if (pNextNode == NULL)
	{
		perror("SnakeMove error");
		exit(1);
	}
	switch (ps->_dir)
	{
	case UP:
		pNextNode->x = ps->_pSnake->x;
		pNextNode->y = ps->_pSnake->y - 1;
		break;
	case DOWN:
		pNextNode->x = ps->_pSnake->x;
		pNextNode->y = ps->_pSnake->y + 1;
		break;
	case LEFT:
		pNextNode->x = ps->_pSnake->x - 2;
		pNextNode->y = ps->_pSnake->y;
		break;
	case RIGHT:
		pNextNode->x = ps->_pSnake->x + 2;
		pNextNode->y = ps->_pSnake->y;
		break;
	}
	if (NextIsFood(pNextNode,ps))
	{
		EatFood(pNextNode, ps);
	}
	else
	{
		NoFood(pNextNode, ps);
	}
	KillByWall(ps);//被墙杀死
	KillBySelf(ps);//被自己杀死
}

//判断是否吃到食物
int NextIsFood(pSnakeNode pn, pSnake ps)
{
	return (ps->_pFood->x == pn->x && ps->_pFood->y == pn->y);
}

//实现贪吃蛇吃食物并增长蛇身
void EatFood(pSnakeNode pn, pSnake ps)
{
	ps->_pFood->next = ps->_pSnake;
	ps->_pSnake = ps->_pFood;
	free(pn);
	pn = NULL;
	pSnakeNode cur = ps->_pSnake;
	while (cur!=NULL)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"●");
		cur = cur->next;
	}
	ps->_score += ps->_food_weight;
	CreateFood(ps);
}

//吃到食物后使食物消失
void NoFood(pSnakeNode pn, pSnake ps)
{
	pn->next = ps->_pSnake;
	ps->_pSnake = pn;
	pSnakeNode  cur = ps->_pSnake;
	while (cur->next->next != NULL)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"●");
		cur = cur->next;
	}
	SetPos(cur->next->x, cur->next->y);
	printf("  ");
	free(cur->next);
	cur->next = NULL;
}

//被墙杀死
void KillByWall(pSnake ps)
{
	if (ps->_pSnake->x == 0 || ps->_pSnake->x == 56 || ps->_pSnake->y == 0 || ps->_pSnake->y == 26)
	{
		ps->_status = KILL_BY_WALL;
	}
}

//被自己杀死
void KillBySelf(pSnake ps)
{
	pSnakeNode cur = ps->_pSnake->next;
	while (cur)
	{
		if (cur->x == ps->_pSnake->x && cur->y == ps->_pSnake->y)
		{
			ps->_status = KILL_BY_SELF;
			break;
		}
		cur = cur->next;
	}
}

//游戏结束
void GameEnd(pSnake ps)
{
	//判断哪种结束方式
	switch (ps->_status)
	{
	case END_NORMAL:
		SetPos(17, 12);
		printf("You have ended the game.");
		break;
	case KILL_BY_WALL:
		SetPos(10, 12);
		printf("You ended the game by hitting a wall.");
		break;
	case KILL_BY_SELF:
		SetPos(10, 12);
		printf("You ended the game by self-collision.");
		break;
	}
	//清除贪吃蛇
	pSnakeNode cur = ps->_pSnake;
	pSnakeNode prev = NULL;
	while (cur)
	{
		prev = cur;
		cur = cur->next;
		free(prev);
	}

}

<test.c>文件

#include "snake.h"
//游戏的主体进程
void test()
{
	char ch;
	do
	{
		system("cls");
		Snake snake = { 0 };
		GameStart(&snake);//游戏开始
		GameRun(&snake);//游戏运行
		GameEnd(&snake);//游戏结束
		SetPos(20, 15);//结束之后,询问是否再来一次
		printf("Play again? (Y/N)");
		ch = getchar();//用户输入一个字符并按回车后,实际上有两个字符进入了输入缓冲区:用户输入的字符和随后的换行符。第一个 getchar() 会读取用户输入的字符,而第二个 getchar() 则用来读取(并丢弃)换行符。
		getchar();
	} while (ch == 'Y'|| ch == 'y');
	SetPos(0, 28);//如果游戏结束,(为了美观)退出代码定位
}

//主函数
int main()
{
	//设置本地环境
	setlocale(LC_ALL, "");
	//生成随机值
	srand((unsigned int)time(NULL));
	//测试游戏
	test();
	return 0;
}

致谢

  感谢您花时间阅读这篇文章!如果您对本文有任何疑问、建议或是想要分享您的看法,请不要犹豫,在评论区留下您的宝贵意见。每一次互动都是我前进的动力,您的支持是我最大的鼓励。期待与您的交流,让我们共同成长,探索技术世界的无限可能!

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

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

相关文章

文件完整性监控:如何提高企业的数据安全性

企业网络庞大而复杂&#xff0c;需要处理大量关键业务数据&#xff0c;这些敏感文件在企业网络中不断传输&#xff0c;并由多个用户和实体存储、共享和访问。FIM 工具或具有 FIM 功能的 SIEM 解决方案使企业能够跟踪未经授权的文件更改、对敏感信息的恶意访问、数据篡改尝试和内…

ubuntu下实时查看CPU,内存(Mem)和GPU的利用率

一、实时查看CPU和内存&#xff08;Mem&#xff09;利用率 htop官网&#xff1a;htop - an interactive process viewer sudo apt-get install htop htop ①. 顶部状态栏&#xff08;System Metrics Overview&#xff09; 这个区域显示系统的全局资源使用情况&#xff0c;包括…

JavaSE——集合12:Map接口实现类—Properties

目录 一、Properties基本介绍 二、Properties常用方法 一、Properties基本介绍 Properties类继承自HashTable类并且实现了Map接口&#xff0c;也是使用一种键值对的形式&#xff0c;来保存数据。Properties的使用特点和HashTable类似Properties还可以用于从xxx.properties文件…

【实践】快速学会使用阿里云消息队列RabbitMQ版

文章目录 1、场景简介2、实验架构和流程2.1、实验架构2.2、实验流程 3、创建实验资源4、创建阿里云AccessKey5、创建静态用户名密码6、创建Vhost、Exchange、Queue并绑定关系6.1、Vhost 的作用6.2、创建Vhost6.3、Exchange 的作用6.4、创建Exchange6.5、Queue 的作用6.6、创建Q…

基于Python flask的豆瓣电影可视化系统,豆瓣电影爬虫系统

博主介绍&#xff1a;✌Java徐师兄、7年大厂程序员经历。全网粉丝13w、csdn博客专家、掘金/华为云等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专栏推荐订阅&#x1f447;&#x1f3fb; 不…

Mysql(七) --- 索引

文章目录 前言1.简介1.1.索引是什么&#xff1f;1.2.为什么使用索引? 2.索引应该使用什么数据结构&#xff1f;2.1.Hash2.2.二叉搜索树2.3.N叉树2.4.B树2.4.1. 简介2.4.2. B树的特点2.4.3. B树和B树的对比 3.Mysql中的页3.1.为什么要使用页3.2.页文件头和页文件尾3.3.页主体3.…

【Linux】解锁线程基本概念和线程控制,步入多线程学习的大门

目录 1、线程初识 1.1线程的概念 1.2.关于线程和进程的进一步理解 1.3.线程的设计理念 1.4.进程vs线程&#xff08;图解&#xff09; 1.5地址空间的第四谈 2.线程的控制&#xff1a; 2.1.关于线程控制的前置知识 2.2创建线程的系统调用&#xff1a; 这个几号手册具体…

JavaScript | 定时器(setInterval和clearInterval)的使用

效果图如下&#xff1a; 当用户第一次看到这个页面时&#xff0c;按钮是不可点击的&#xff0c;并显示一个5秒的倒计时。倒计时结束后&#xff0c;按钮变为可点击状态&#xff0c;并显示“同意协议”。这样做的目的是确保用户有足够的时间阅读用户协议。 <!DOCTYPE html>…

机器学习:知识蒸馏(Knowledge Distillation,KD)

知识蒸馏&#xff08;Knowledge Distillation&#xff0c;KD&#xff09;作为深度学习领域中的一种模型压缩技术&#xff0c;主要用于将大规模、复杂的神经网络模型&#xff08;即教师模型&#xff09;压缩为较小的、轻量化的模型&#xff08;即学生模型&#xff09;。在实际应…

Vue(3) 组件

文章目录 对组件的理解单文件组件非单文件组件基本使用几个注意点组件的嵌套VueComponent构造函数一个重要的内置关系 组件的自定义事件全局事件总线安装全局事件总线使用事件总线解绑事件消息订阅与发布简介使用步骤范例 $nextTick插槽1.默认插槽2.具名插槽作用域插槽 对组件的…

[linux 驱动]网络设备驱动详解

目录 1 描述 2 结构体 2.1 net_device 2.2 sk_buff 2.3 net_device_ops 2.4 ethtool_ops 3 相关函数 3.1 网络协议接口层 3.1.1 dev_queue_xmit 3.1.2 netif_rx 3.1.3 alloc_skb 3.1.4 kfree_skb 3.1.5 skb_put 3.1.6 skb_push 3.1.7 skb_reserve 3.2 网络设备驱…

使用OpenCV实现基于EigenFaces的人脸识别

引言 人脸识别技术近年来得到了飞速的发展&#xff0c;它被广泛应用于安全监控、门禁系统、智能设备等领域。其中&#xff0c;基于特征脸&#xff08;EigenFaces&#xff09;的方法是最早期且较为经典的人脸识别算法之一。本文将介绍如何使用Python和OpenCV库实现一个简单的人…

AI大模型面经——以医疗领域为例,整理RAG基础与实际应用中的痛点

前言 谈到大模型在各垂直领域中的应用&#xff0c;一定离不开RAG&#xff0c;本系列开始分享一些RAG相关使用经验&#xff0c;可以帮助大家在效果不理想的时候找到方向排查或者优化。 本系列以医疗领域为例&#xff0c;用面试题的形式讲解RAG相关知识&#xff0c;开始RAG系列…

Spring与Spring Boot之间的区别

Spring和Spring Boot是用于开发Java企业应用的两个主流框架。虽然它们都属于Spring生态系统的一部分&#xff0c;但是它们各自有不同的使用场景和特点。 在本文中&#xff0c;我们将探讨Spring与Spring Boot之间的差异&#xff0c;针对他们之间特性的差异&#xff0c;做一个详…

windows10系统-在线设置账户有效期

.a.打开微软官网 点击右上角的登录按钮 .b.输入自己idea账户名&#xff0c;然后点击下一步 .c.使用邮箱接收验证码 输入验证码->登录 .d.点击右上角的个人账户->我的 Microsoft 账户 .e.点击更改密码即可 .f.修改完密码之后&#xff0c;点击保存即可。 微软的在线账户和…

XGBoost回归预测 | MATLAB实现XGBoost极限梯度提升树多输入单输出

回归预测 | MATLAB实现XGBoost极限梯度提升树多输入单输出 目录 回归预测 | MATLAB实现XGBoost极限梯度提升树多输入单输出预测效果基本介绍模型描述程序设计参考资料预测效果 基本介绍 XGBoost的全称是eXtreme Gradient Boosting,它是经过优化的分布式梯度提升库,旨在高效、…

【python学习】1-1 python软件安装

1。搜索python官网&#xff0c;点击下载进行下载安装包。 2.双击.exe文件&#xff0c;如图步骤 3.winR输入cmd&#xff0c;在弹出黑框中输入python显示如图安装好。

linux线程 | 同步与互斥(上)

前言&#xff1a;本节内容主要是线程的同步与互斥。 本篇文章的主要内容都在讲解互斥的相关以及周边的知识。大体的讲解思路是通过数据不一致问题引出锁。 然后谈锁的使用以及申请锁释放锁的原子性问题。 那么&#xff0c; 废话不多说&#xff0c; 现在开始我们的学习吧&#x…

基于element-ui的upload组件与阿里云oss对象存储的文件上传(采用服务端签名后直传的方式)

服务端签名后直传图解 步骤 1 开通阿里云OSS对象存储服务&#xff0c;创建新的Bucket 2 创建子账户获取密钥 创建用户 添加权限 后端 1 新建一个第三方服务的模块 third-party pom文件 <?xml version"1.0" encoding"UTF-8"?> <project x…

【工程测试技术】第4章 常用传感器分类,机械式,电阻式,电容式,电感式,光电式传感器

上理考研周导师的哔哩哔哩频道 我在频道里讲课哦 目录 4.1 常用传感器分类 4.2 机械式传感器及仪器 4.3 电阻式、电容式与电感式传感器 1.变阻器式传感器 2.电阻应变式传感器 3.固态压阻式传感器 4.典型动态电阻应变仪 4.3.2 电容式传感器 1.变换原理 2.测量电路 …