贪吃蛇游戏(代码篇)

news2025/1/16 13:56:24

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

前言 

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

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

有关贪吃蛇必备知识的小项目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;
}

致谢

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

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

使用Java API访问Apache Kafka

简介 Kafka是由Apache软件基金会开发的一个开源流处理平台,Kafka是一个功能强大且灵活的平台。 基本情况 软件名称:Apache Kafka 软件平台:跨平台 软件语言:Scala、Java 开发商:Apache软件基金会 软件授权:Apache License 2.0 最近更新时间:2024年7月23日 核心概念 -…

Ubuntu:用户不在sudoers文件中

1、问题 执行sudo xxx命令时&#xff0c;显示&#xff1a; user 不在sudoers文件中 需要查看系统版本进入恢复模式修复。 2、重启进入恢复模式 查看系统命令&#xff1a;uname -r 可能显示为&#xff1a;6.8.0-45-generic 重启Ubuntu系统&#xff0c;在开机时按ESC进入模…

oracle归档日志爆满问题处理

最近客户单位的oracle数据库出了问题&#xff0c;经常出现无法连接,报错提示 ORA-00257: archiver error, Connect internal only, until freed.&#xff0c;手动清除归档日志后可以恢复访问&#xff0c;但是过不了几天依旧会爆满&#xff0c;每日生成的归档日志很大。经过详细…

内部排序算法小结

练习目标&#xff1a; 1、实现 直接插入排序、冒泡排序、SHELL排序和快速排序&#xff1b; 2、随机生成100组数据&#xff0c;每组数据1000个元素。 注意&#xff1a;计时的单位是CPU的clock而非时间&#xff01;&#xff01;&#xff01; 【后续】 1、加入选择排序&#xff1b…

读书笔记《PPT演讲力》大树模型

作者把PPT演讲比作一棵大树&#xff0c;树的每一部分对应着PPT演讲的一个技巧。 根据这个大树模型&#xff0c;是否有联想到自己过往的演讲经历&#xff1f;演讲是否都达到了大树模型中说的效果&#xff1f;根据这个思维导图&#xff0c;结合自己的经历&#xff0c;试着总结3句…

云计算第四阶段-----CLOUND二周目 04-06

cloud 04 今日目标&#xff1a; 一、Pod 生命周期 图解&#xff1a; [rootmaster ~]# vim web1.yaml --- kind: Pod apiVersion: v1 metadata:name: web1 spec:initContainers: # 定义初始化任务- name: task1 # 如果初始化任务失败&#…

数字化与数智化,你知道它们的区别吗?

​其实老早就想说这个。中间一直在忙忙忙&#xff0c;还有处理自己的事情&#xff0c;导致拖更了。 最近听说一个物流大佬现在也转行做数字化厂家负责人&#xff0c;顺便给我讲解了这二者的区别&#xff0c;这里我就重新梳理了一下&#xff0c;加上了我自己的一些观点&#xf…

qt creator 开发环境的安装

1.找官网 官网地址&#xff1a;Installation | Qt Creator Documentation 点 Parent Directory 继续点 Parent Directory 点 archive/ 2.下载在线安装器 点 online_ainstallers 选择在线安装器版本 选择对应版本后进入下载列表&#xff0c;根据自己的系统选择下载。 下载后…

常用类(四)---String结构剖析

文章目录 1.String结构剖析2.String创建剖析3.String特性分析4.String方法总结5.StringBuffer和StringBuilder总结5.1stringbuffer对象的创建方法5.2string--stringbuffer之间的相互转换&#xff1a; 这个是我第二次学习string&#xff0c;听的是hsp老师的课程&#xff0c;我认…

IRP读写函数

驱动代码&#xff1a; #include <ntddk.h>#define DEVICE_NAME L"\\device\\MyDricer1" //设备对象名称 #define LINK_NAME L"\\dosdevices\\Goose" //符号链接名称VOID UnDirver(PDRIVER_OBJECT pDriverObj) {UNICODE_STRING uLinkName RTL_CONST…

OpenCV高级图形用户界面(6)获取指定窗口中图像的矩形区域函数getWindowImageRect()的使用

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 提供窗口中图像的矩形区域。 该函数 getWindowImageRect 返回图像渲染区域的客户端屏幕坐标、宽度和高度。 函数原型 Rect cv::getWindowImage…

旺店通ERP集成金蝶云星瀚(旺店通主供应链)

源系统成集云目标系统 金蝶云星瀚介绍 金蝶云星瀚是专为大企业设计的新一代数字化管理云服务、大型企业SaaS管理云&#xff0c;旨在凭借千亿级高性能和世界一流企业的实践&#xff0c;帮助大企业实现可信的数字化系统升迁&#xff0c;打造韧性企业&#xff0c;支撑商业创新和管…

C++:list(用法篇+模拟实现)

文章目录 前言一、list 的用法1. list 简介2. 用法代码演示1&#xff09;头/尾 插/删和迭代器遍历2&#xff09;insert与erase3&#xff09;排序sort相关4&#xff09;其他相关 二、list模拟实现1. 结点类模板list_node2. 定义迭代器1&#xff09;为什么要专门封装一个迭代器&a…

【uniapp】设置公共样式,实现公共背景等

目录 1、 全局渐变背景色 2.1 创建common目录 2.2 在common下新建style和images等目录 2.3 在style下新建common-style.scss 2.4 common-style输入全局渐变颜色 2.5 引入样式 2.6 业务页面引入 2.7 展示 2、全局字体颜色 2.1 新建base-style.scss文件 2.2 设置base-…

STM32的GPIO寄存器描述

寄存器&#xff1a; 软件控制硬件(在程序中操作对应控制器)&#xff0c;通过寄存器&#xff0c;就是 寄存器(可以存放数据)&#xff0c;但是其中的数据具有特定的硬件含义(查看芯片手册)&#xff0c;设置寄存器的值&#xff0c;对应的控制器就执行对应的工作。相当于寄存器就是…

IntelliJ IDEA中配置scala

1.IDEA中 配置 maven 左上角 file -> Setting 选择(或直接搜maven) Build, Execution,Deployment -> Build Toos -> Maven Maven home path 选择 maven 安装目录&#xff08;bin的上层目录&#xff09; 示例&#xff1a; D:\maven\apache-maven-3.8.6 User settings…

2024.10月11日--- SpringMVC拦截器

拦截器 1 回顾过滤器&#xff1a; Servlet规范中的三大接口&#xff1a;Servlet接口&#xff0c;Filter接口、Listener接口。 过滤器接口&#xff0c;是Servlet2.3版本以来&#xff0c;定义的一种小型的&#xff0c;可插拔的Web组件&#xff0c;可以用来拦截和处理Servlet容…

基于Springboot+Vue的租房管理系统 (含源码数据库)

1.开发环境 开发系统:Windows10/11 架构模式:MVC/前后端分离 JDK版本: Java JDK1.8 开发工具:IDEA 数据库版本: mysql5.7或8.0 数据库可视化工具: navicat 服务器: SpringBoot自带 apache tomcat 主要技术: Java,Springboot,mybatis,mysql,vue 2.视频演示地址 3.功能 系统根…

yjs机器学习常见算法01——KNN(K—近邻算法)

1.K—近邻算法 的含义&#xff1a; 简单来说就是通过你的邻居的“类别”&#xff0c;来推测你的“类别” 定义&#xff1a;如果一个样本在特征空间中的k个最相似&#xff08;即特征空间中最临近&#xff09;的样本中大多数属于某一类别&#xff0c;则该样本也属于这个类别。 2.…