【贪吃蛇:C语言实现】

news2025/1/19 11:14:29

文章目录

  • 前言
  • 1.了解Win32API相关知识
    • 1.1什么是Win32API
    • 1.2设置控制台的大小、名称
    • 1.3控制台上的光标
    • 1.4 GetStdHandle(获得控制台信息)
    • 1.5 SetConsoleCursorPosition(设置光标位置)
    • 1.6 GetConsoleCursorInfo(获得光标信息)
    • 1.7 SetConsoleCursorInfo(设置光标信息)
    • 1.8 GetAsyncKeyState(获取按键信息)
  • 2.游戏设计与分析
    • 2.1地图设计
    • 2.2数据结构的设计
      • 2.2.1蛇身的设计
      • 2.2.2蛇的方向
      • 2.2.3游戏的状态
      • 2.2.4管理整条蛇
    • 2.3.游戏流程
  • 3. 功能的具体实现
    • 游戏开始前
    • 3.1 初始化蛇
    • 3.2 创建食物
    • 游戏运行
    • 3.3 检测按键
    • 3.4 蛇移动
    • 3.5 判断节点是否是食物
    • 3.6 吃食物
    • 3.7 不吃食物
    • 3.8 撞墙
    • 3.9 撞到自己
    • 3.10 游戏结束
    • 3.11 main函数
  • 4.参考代码
    • 4.1 main
    • 4.2 Snake.h
    • 4.3 Snake.c

在这里插入图片描述

前言

彻底学习完C语言之后,为巩固所学,我们来实现一个贪吃蛇小游戏。
该游戏涉及到的C语言的知识有:函数、枚举、结构体、动态内存管理、预处理指令。
另外,还包含数据结构(链表)以及Win32API相关知识。

先看一下游戏效果吧
在这里插入图片描述

目前该游戏还有很多可以改进的地方,由于蛇的移动使用的是Sleep函数,看起来有点卡顿,而且按键会有一点延迟。
代码仓库链接

1.了解Win32API相关知识

1.1什么是Win32API

Win32 API也就是Microsoft Windows32位平台的应用程序编程接口。调用各种接口,可以帮应⽤程序达到开启视窗、描绘图形、使⽤周边设备等目的,由于这些函数服务的对象是应用程序(Application), 所以便称之为 Application Programming Interface,简称 API 函数。

1.2设置控制台的大小、名称

设置大小和名称的命令为:

  • mode con cols=列 lines=行
  • tiele 名称

如果想在C语言中设置窗口,需要使用system函数。

在这里插入图片描述

1.3控制台上的光标

COORD 是Windows API中定义的⼀个结构体,表示⼀个字符在控制台屏幕上的坐标,坐标系(0,0)的原点位于缓冲区的左侧顶部的单元格。使用需要window.h的头文件
在这里插入图片描述

COORD类型的声明:

typedef struct _COORD {
	SHORT X;//横坐标
	SHORT Y;//纵坐标
} COOR, *PCOORD;

1.4 GetStdHandle(获得控制台信息)

GetStdHandle是⼀个Windows API函数。它⽤于从⼀个特定的标准设备(标准输⼊、标准输出或标准错误)中取得⼀个句柄(⽤来标识不同设备的数值),使用这个句柄可以操作设备(控制台)。

HANDLE WINAPI GetStdHandle( _In_ DWORD nStdHandle);

该函数需要一个参数,参数如下:
在这里插入图片描述

1.5 SetConsoleCursorPosition(设置光标位置)

设置指定控制台屏幕缓冲区中的光标位置,我们将想要设置的坐标信息放在COORD类型的pos中,调用SetConsoleCursorPosition函数将光标位置设置到指定的位置。

BOOL WINAPI SetConsoleCursorPosition(
 	HANDLE hConsoleOutput,     /要设置哪个控制台
 	COORD pos		           /将光标设置的位置
);

通过上述三个功能,实现设置光标位置

在这里插入图片描述

1.6 GetConsoleCursorInfo(获得光标信息)

GetConsoleCursorInfo:检索有关指定控制台屏幕缓冲区的光标大小可见性的信息。然后将获得的信息放在一个结构体中

BOOL WINAPI GetConsoleCursorInfo(
  _In_  HANDLE      hConsoleOutput,
  _Out_ PCONSOLE_CURSOR_INFO   lpConsoleCursorInfo
);

_ Out_ PCONSOLE_CURSOR_INFO 是指向 CONSOLE_CURSOR_INFO 结构体的指针

该结构体,包含有关控制台光标的信息

typedef struct _CONSOLE_CURSOR_INFO {
 	DWORD dwSize;	
 	BOOL bVisible;	
} CONSOLE_CURSOR_INFO, *PCONSOLE_CURSOR_INFO;
  • dwSize,由光标填充的字符单元格的百分比。 此值介于1到100之间。 光标外观会变化,范围从完全填充单元格到单元底部的⽔平线条。
  • bVisible,光标的可见性。 如果光标可见,则此成员为 TRUE。

在这里插入图片描述

1.7 SetConsoleCursorInfo(设置光标信息)

SetConsoleCursorInfo:设置指定控制台屏幕缓冲区的光标的大小和可见性。
其参数和get方法相同

BOOL WINAPI SetConsoleCursorInfo(
 	HANDLE hConsoleOutput,
 	const CONSOLE_CURSOR_INFO *lpConsoleCursorInfo
);

设置前:

在这里插入图片描述
设置后:

在这里插入图片描述
隐藏光标:

在这里插入图片描述

1.8 GetAsyncKeyState(获取按键信息)

GetAsyncKeyState:获取按键情况,GetAsyncKeyState的函数原型如下:

SHORT GetAsyncKeyState(
 	int vKey
);
  • 将键盘上每个键的虚拟键值传递给函数,函数通过返回值来分辨按键的状态。了解虚拟键值点击这里
  • GetAsyncKeyState 的返回值是short类型,在上⼀次调⽤ GetAsyncKeyState 函数后,如果返回的16位的short数据中,最高位是1,说明按键的状态是按下,如果最⾼是0,说明按键的状态是抬起如果最低位被置为1则说明,该按键被按过,否则为0
  • 如果我们要判断⼀个键是否被按过,可以检测GetAsyncKeyState返回值的最低值是否为1
 #define KEY_PRESS(VK) ( (GetAsyncKeyState(VK) & 0x1) ? 1 : 0 )
#include<Windows.h>
int main()
{
	while (1)
	{
		if (KEY_PRESS(0x30))
			printf("0\n");
		else if (KEY_PRESS(0x31))
			printf("1\n");
		else if (KEY_PRESS(0x32))
			printf("2\n");
		else if (KEY_PRESS(0x33))
			printf("3\n");
		else if (KEY_PRESS(0x34))
			printf("4\n");
		else if (KEY_PRESS(0x35))
			printf("5\n");
		else if (KEY_PRESS(0x36))
			printf("6\n");
		else if (KEY_PRESS(0x37))
			printf("7\n");
	}
	return 0;
}

2.游戏设计与分析

2.1地图设计

由于我打印墙体和蛇想使用 □,但是它是宽字符,该如何打印呢?
在这里插入图片描述
C语言规定:宽字符的类型wchar_t 和宽字符的输⼊和输出函数。
头文件为:<locale.h>

在这里插入图片描述

宽字符与普通字符对比:

在这里插入图片描述

由于墙体使用的是宽字符,所以在打印时需要注意!

在这里插入图片描述
在这里插入图片描述

以下函数实现打印墙体的功能:

#define WALL L'□'
void DrawMap()
{
	SetPos(0, 0);
	//上
	for (int i = 0; i <= 56; i += 2)
		wprintf(L"%lc", WALL);
	//左、右
	for (int i = 1; i <= 25; i++)
	{
		SetPos(0, i);
		wprintf(L"%lc", WALL);
		SetPos(56, i);
		wprintf(L"%lc", WALL);
	}
	//下
	SetPos(0, 26);
	for (int i = 0; i <= 56; i += 2)
		wprintf(L"%lc", WALL);  
}

再打印地图之前,我们需要设置窗口的大小,并且给玩家一些引导(Win32API的使用已在上面讲过)

//窗口相关设置
void ScreenPrepare()
{
	//设置窗体大小
	system("mode con cols=100 lines=40");
	system("title 贪吃蛇小游戏");
	//隐藏光标
	HideCurSor();
	//打印欢迎语
	Welcome();
	//打印地图
	DrawMap();
	//显示提示信息
	HelpInfo();
}

具体函数的实现我们会在后面给出,先看一下效果吧~

在这里插入图片描述

2.2数据结构的设计

2.2.1蛇身的设计

在游戏运行的过程中,蛇每次吃⼀个⻝物,蛇的⾝体就会变长⼀节,如果我们使用链表存储蛇的信息,那么蛇的每⼀节其实就是链表的每个节点。每个节点只要记录好蛇身节点在地图上的坐标就行,所以蛇节点结构如下:

//蛇节点的定义
typedef struct SnakeNode
{
	int x;
	int y;
	struct SnakeNode* next;//指向下一个节点
}SnakeNode,*pSnakeNode;//同时定义一个指向该节点的指针

2.2.2蛇的方向

蛇的方向可以一一列举,使用枚举

//蛇的方向
//上、下、做、左、右
enum Direction
{
	UP = 1,
	DOWN,
	LEFT,
	RIGHT
};

2.2.3游戏的状态

游戏状态,可以⼀⼀列举,使用枚举

//游戏的状态
enum GameState
{
	OK,//正常
	KILL_BY_WALL,//撞墙
	KILL_BY_SELF,//撞到自己
	END//结束
};

2.2.4管理整条蛇

我们再封装⼀个Snake的结构体来维护整条贪吃蛇。

//整条蛇
typedef struct Snake
{
	pSnakeNode pSnake;  //指向整条蛇的指针
	pSnakeNode PFood;   //指向食物的指针

	enum Direction Dire;  //蛇的方向
	enum GameState State; //蛇的状态

	int SleepTime; //蛇的休眠时间;休眠越短,蛇移动的越快
	int Score;//游戏得分
	int FoodScore;//一个食物的分数
}Snake,* pSnake;

2.3.游戏流程

在这里插入图片描述

3. 功能的具体实现

游戏开始前

3.1 初始化蛇

游戏开始,蛇的长度默认是3。

  • 创建节点,头插法依次连接
  • 节点的纵坐标相同,横坐标依次增加2,产生串的效果
  • 蛇的方向默认为右
  • 打印蛇,并且为蛇头设置颜色

此处颜色设置我们使用Win32API提供的函数:

BOOL WINAPI SetConsoleTextAttribute(
  _In_ HANDLE hConsoleOutput,//当前设备句柄
  _In_ WORD   wAttributes//颜色
);

在这里插入图片描述

//设置颜色
void SetColor(int HeadColor)
{
	SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE), HeadColor);
}

//打印蛇
void PrintSnake(pSnake ps)
{
	pSnakeNode cur = ps->pSnake;
	int flag = 1;
	while (cur)
	{
		if (flag == 1)
		{
			//为蛇头设置颜色
			SetColor(12);
		}
		SetPos(cur->x, cur->y);//设置每个节点的位置
		wprintf(L"%lc", BODY);
		cur = cur->next;
		flag = 0;
		SetColor(10);
	}
}

//初始化蛇
void InitSnake(pSnake ps)
{
	int i = 0;
	pSnakeNode cur = NULL;
	//蛇的长度开始为3
	for (i = 0; i < 3; i++)
	{
		cur = (pSnakeNode)malloc(sizeof(SnakeNode));
		if (cur == NULL)
		{
			perror("InitSnake():malloc");
			return;
		}
		cur->next = NULL;
		cur->x = POS_X + i * 2;  //节点横坐标依次增加2
		cur->y = POS_Y;

		//头插法将节点相连
		if (ps->pSnake == NULL)
		{
			ps->pSnake = cur;
		}
		else
		{
			cur->next = ps->pSnake;
			ps->pSnake = cur;
		}
	}
	//打印蛇
	PrintSnake(ps);
	//初始化蛇的其它信息
	ps->Dire = RIGHT;
	ps->FoodScore = 10;
	ps->SleepTime = 200;
	ps->State = OK;
	ps->Score = 0;
}

3.2 创建食物

食物要随机生成,有以下几个注意事项:

  • 食物应该在墙体内
  • 食物不可与蛇身重叠
  • 由于我们蛇的打印使用的是宽字符,所以食物的坐标应该在2的倍数处
//创建食物
void CreatFood(pSnake ps)
{
	int x = 0;
	int y = 0;
	//食物坐标应该在墙体内
again:
	do
	{
		x = rand() % 53 + 2;
		y = rand() % 25 + 1;
	} while (x % 2 != 0);

	//食物坐标不与蛇重叠
	pSnakeNode cur = ps->pSnake;
	while (cur)
	{
		if (x == cur->x && y == cur->y)
		{
			goto again;
		}
		cur = cur->next;
	}
	//根据坐标创建食物
	pSnakeNode pFood = (pSnakeNode)malloc(sizeof(SnakeNode));
	if (pFood == NULL)
	{
		perror("CreatFood():malloc()");
		return;
	}
	//打印食物
	pFood->x = x;
	pFood->y = y;
	SetColor(14);
	SetPos(pFood->x, pFood->y);
	wprintf(L"%lc", FOOD);
	SetColor(10);
	ps->pFood = pFood;
}

在这里插入图片描述

游戏运行

3.3 检测按键

玩家按上下左右键控制蛇的移动,在检测时需要注意:

  • 玩家按的键若与蛇的方向相反,则不做响应;玩家按键符合,则修改蛇的方向
  • 玩家按F1或F2修改蛇的移动速度;蛇的速度应大于0,食物的分数最低为1分
  • 按键的检测使用Win32API提供的功能:GetAsyncKeyState
//玩游戏
void GameRun(pSnake ps)
{
	do
	{
		//显示分数
		SetColor(12);
		SetPos(65, 12);
		printf("目前得分:%-5d",ps->Score);
		SetColor(10);
		SetPos(65, 13);
		printf("每个食物:%2d分",ps->FoodScore);
		//检测按键
		if (KEY_PRESS(VK_UP) && ps->Dire != DOWN)
		{
			//按上键,且蛇的方向不能向下
			ps->Dire = UP;
		}
		else if (KEY_PRESS(VK_DOWN) && ps->Dire != UP)
		{
			//按下键,且蛇的方向不能向上
			ps->Dire = DOWN;
		}
		else if (KEY_PRESS(VK_LEFT) && ps->Dire != RIGHT)
		{
			//按左键,且蛇的方向不能向右
			ps->Dire = LEFT;
		}
		else if (KEY_PRESS(VK_RIGHT) && ps->Dire != LEFT)
		{
			//按右键,且蛇的方向不能向左
			ps->Dire = RIGHT;
		}
		else if (KEY_PRESS(VK_ESCAPE))
		{
			//按ESC,退出游戏
			ps->State = END;
		}
		else if (KEY_PRESS(VK_SPACE))
		{
			//按空格,暂停
			pause();
		}
		else if (KEY_PRESS(VK_F1))
		{
			//按F1加速,即睡眠时间变短
			
			//休眠时间不能是负数,最快就是休眠30ms
			if (ps->SleepTime >= 50)
			{
				ps->SleepTime -= 20;
				//速度变快,食物的分数变高
				ps->FoodScore += 2;
			}
		}
		else if (KEY_PRESS(VK_F2))
		{
			//F2减速,睡眠时间变长

			//食物的分数不能减到负数,最多减为1分
			if (ps->FoodScore >= 3)
			{
				ps->SleepTime += 20;
				ps->FoodScore -= 2;
			}
		}
		//按照蛇的睡眠时间,真正实现休眠
		Sleep(ps->SleepTime);

		//休眠后,蛇要移动
		SnakeMove(ps);

	} while (ps->State == OK);//只有蛇的状态为OK,才能继续检测按键
}

3.4 蛇移动

蛇的移动就是根据蛇的方向,产生节点,判断节点是不是食物。

  • 所产生的节点的位置根据蛇的方向而定
  • 节点是食物,吃掉食物,蛇身变长
  • 节点不是食物,也吃掉食物,但长度不变
//移动蛇
void SnakeMove(pSnake ps)
{
	//先产生节点
	pSnakeNode pNext = (pSnakeNode)malloc(sizeof(SnakeNode));
	if (pNext == NULL)
	{
		perror("SnakeMove():malloc()");
		return;
	}
	//根据蛇的方向,设定节点的位置
	switch (ps->Dire)
	{
	case UP:
		//蛇的方向向上,节点应该在当前蛇头的上方;横坐标不变,纵坐标减1
		pNext->x = ps->pSnake->x;
		pNext->y = ps->pSnake->y - 1;
		break;
	case DOWN:
		//蛇的方向向下,节点应该在当前蛇头的下方;横坐标不变,纵坐标加1
		pNext->x = ps->pSnake->x;
		pNext->y = ps->pSnake->y + 1;
		break;
	case LEFT:
		//蛇的方向向左,节点应该在当前蛇头的左边;纵坐标不变,横坐标减2
		pNext->x = ps->pSnake->x - 2;
		pNext->y = ps->pSnake->y;
		break;
	case RIGHT:
		//蛇的方向向右,节点应该在当前蛇头的右方;纵坐标不变,横坐标加2
		pNext->x = ps->pSnake->x + 2;
		pNext->y = ps->pSnake->y;
		break;
	}

	//判断下一个节点是否是食物
	if (JudgeNext(ps,pNext))
	{
		//是食物,吃掉,长度增加
		EatFood(ps, pNext);
	}
	else
	{
		//不是食物,吃掉,长度不增加
		NoFood(ps, pNext);
	}

	//未完
	//撞墙
	
	//撞自己
}

3.5 判断节点是否是食物

将所产生的节点与食物的节点对比即可

//判断节点是否是食物
int JudgeNext(pSnake ps, pSnakeNode pNext)
{
	return ((ps->pFood->x == pNext->x) && (ps->pFood->y == pNext->y));
}

3.6 吃食物

吃掉食物很简单,将所产生的节点与蛇想连;连接后得分增加,并释放所产生的节点,再次产生食物。

//是食物,吃掉,长度增加
void EatFood(pSnake ps, pSnakeNode pNext)
{
	//吃掉食物,头插法将节点插入
	pNext->next = ps;
	ps = pNext;
	//打印蛇
	PrintSnake(ps_->snake);
	//加分
	PrintSnake(ps->pSnake);
	//释放食物节点
	free(ps->pFood);
	//再次创建食物
	CreatFood(ps);
}

3.7 不吃食物

不吃食物需要将产生的节点与蛇相连,然后删除蛇尾(将蛇尾打印尾空格,并释放蛇尾节点)

//不吃食物
void NoFood(pSnake ps, pSnakeNode pNext)
{
	//头插法连接
	pNext->next = ps->pSnake;
	ps->pSnake = pNext;
	//删除蛇尾
	pSnakeNode cur = ps->pSnake;
	//找到蛇尾的前一个节点
	while (cur->next->next)
	{
		cur = cur->next;
	}
	pSnakeNode del = cur->next;
	//将蛇尾打印尾空格
	SetPos(del->x, del->y);
	printf("  ");
	//释放蛇尾节点
	free(del);
	cur->next = NULL;
	//打印蛇
	PrintSnake(ps);
}

3.8 撞墙

若蛇头坐标与墙体坐标重合,则说明撞墙了。

//撞墙
void KillByWall(pSnake ps)
{
	if (ps->pSnake->x == 0 ||
		ps->pSnake->x == 56 ||
		ps->pSnake->y == 0 ||
		ps->pSnake->y == 26)
	{
		ps->State = KillByWall;
	}
}

3.9 撞到自己

若蛇头坐标与蛇身节点的坐标重合,则说明撞到自己了。

//撞到自己
void KillBySelf(pSnake ps)
{
	pSnakeNode cur = ps->pSnake->next;
	while (cur)
	{
		if (cur->x == ps->pSnake->x && cur->y == ps->pSnake->y)
		{
			ps->State = KILL_BY_SELF;
		}
		cur = cur->next;
	}
}

3.10 游戏结束

给出游戏结束的原因,释放蛇身节点。

//游戏结束
void GameEnd(pSnake ps)
{
	SetPos(20, 13);
	SetColor(12);

	switch (ps->State)
	{
	case END:
		printf("您结束了游戏");
		break;
	case KILL_BY_SELF:
		printf("很遗憾!您撞到了自己");
		break;
	case KILL_BY_WALL:
		printf("很遗憾!您撞墙了");
		break;
	}
	pSnakeNode cur = ps->pSnake;
	while (cur)
	{
		pSnakeNode del = cur;
		cur = cur->next;
		free(del);
	}
	ps = NULL;
	SetColor(10);
}

3.11 main函数

让玩家选择

int main()
{
	srand((unsigned int)time(NULL));
	setlocale(LC_ALL, "");//适应本地中文环境

	int ch = 0;
	do
	{
		Snake snake = { 0 };//创建贪吃蛇
		//游戏开始前的初始化
		GameStart(&snake);

		玩游戏
		GameRun(&snake);

		游戏结束,善后工作
		GameEnd(&snake);
		SetPos(20, 15);
		printf("还要再来一局吗?(Y/N)");
		ch = getchar();
		getchar();//吸收换行
	} while (ch == 'Y' || ch == 'y');

	SetPos(0, 30);
	return 0;
}

4.参考代码

4.1 main

#define _CRT_SECURE_NO_WARNINGS 1
#include"Snake.h"

int main()
{
	srand((unsigned int)time(NULL));
	setlocale(LC_ALL, "");//适应本地中文环境

	int ch = 0;
	do
	{
		Snake snake = { 0 };//创建贪吃蛇
		//游戏开始前的初始化
		GameStart(&snake);

		玩游戏
		GameRun(&snake);

		游戏结束,善后工作
		GameEnd(&snake);
		SetPos(20, 15);
		printf("还要再来一局吗?(Y/N)");
		ch = getchar();
		getchar();//吸收换行
	} while (ch == 'Y' || ch == 'y');

	SetPos(0, 30);
	return 0;
}

4.2 Snake.h

#define _CRT_SECURE_NO_WARNINGS 1

#include<locale.h>
#include<stdio.h>
#include<Windows.h>
#include<stdlib.h>
#include<stdbool.h>
#include<time.h>

#define KEY_PRESS(VK) ((GetAsyncKeyState(VK)&0x1) ? 1 : 0)

#define WALL L'□'//墙体
#define BODY L'●'//蛇身
#define FOOD L'★'//食物

//蛇初始位置
#define POS_X 24
#define POS_Y 12

//蛇的方向
//上、下、左、右
enum Direction
{
	
	UP = 1,
	DOWN,
	LEFT,
	RIGHT
};

//游戏的状态
enum GameState
{
	OK,//正常
	KILL_BY_WALL,//撞墙
	KILL_BY_SELF,//撞到自己
	END//结束
};

//蛇节点的定义
typedef struct SnakeNode
{
	int x;
	int y;
	struct SnakeNode* next;//指向下一个节点
}SnakeNode,*pSnakeNode;//同时定义一个指向该节点的指针

//整条蛇
typedef struct Snake
{
	pSnakeNode pSnake;  //指向整条蛇的指针
	pSnakeNode pFood;   //指向食物的指针

	enum Direction Dire;  //蛇的方向
	enum GameState State; //蛇的状态

	int SleepTime; //蛇的休眠时间;休眠越短,蛇移动的越快
	int Score;//游戏得分
	int FoodScore;//一个食物的分数
}Snake,* pSnake;

//设置光标位置
void SetPos(int x, int y);

//设置颜色
void SetColor(int HeadColor);

//游戏准备
void GameStart(pSnake psnake);

//初始化蛇
void InitSnake(pSnake psnake);

//创建食物
void CreatFood(pSnake ps);

//打印蛇
void PrintSnake(pSnake ps);

//玩游戏
void GameRun(pSnake ps);

//移动蛇
void SnakeMove(pSnake ps);

//判断节点是否是食物
int JudgeNext(pSnake ps, pSnakeNode pNext);

//是食物,吃掉,长度增加
void EatFood(pSnake ps, pSnakeNode pNext);

//不是食物
void NoFood(pSnake ps, pSnakeNode pNext);

//撞墙
void KillByWall(pSnake ps);

//撞到自己
void KillBySelf(pSnake ps);

//游戏结束
void GameEnd(pSnake ps);

4.3 Snake.c

#define _CRT_SECURE_NO_WARNINGS 1

#include"Snake.h"

//隐藏光标
void HideCurSor()
{
	CONSOLE_CURSOR_INFO cursor = { 0 };
	HANDLE handle = GetStdHandle(STD_OUTPUT_HANDLE);
	GetConsoleCursorInfo(handle, &cursor);
	cursor.bVisible = false;
	SetConsoleCursorInfo(handle, &cursor);
}

//设置光标位置
void SetPos(int x, int y)
{
	COORD pos = { x,y }; //要设置的位置
	HANDLE handle = GetStdHandle(STD_OUTPUT_HANDLE);//哪个设备
	SetConsoleCursorPosition(handle, pos);//设置
}

void Welcome()
{
	//显示欢迎语一
	SetPos(38, 15);
	printf("欢迎来到贪吃蛇小游戏!");
	SetPos(40, 24);
	system("pause");
	system("cls");//清屏
	//提示语
	SetPos(25, 15);
	printf("按 ↑、↓、←、→ 控制蛇蛇的移动,F1为加速,F2为减速,");
	SetPos(25, 16);
	printf("加速将能得到更高的分数");
	SetPos(40, 24);
	system("pause");
}

//打印地图
void DrawMap()
{
	system("cls");
	SetColor(6);
	SetPos(0, 0);
	//上
	for (int i = 0; i <= 56; i += 2)
		wprintf(L"%lc", WALL);
	//左、右
	for (int i = 1; i <= 25; i++)
	{
		SetPos(0, i);
		wprintf(L"%lc", WALL);
		SetPos(56, i);
		wprintf(L"%lc", WALL);
	}
	//下
	SetPos(0, 26);
	for (int i = 0; i <= 56; i += 2)
		wprintf(L"%lc", WALL);  
	SetColor(10);
}

//显示提示信息
void HelpInfo()
{
	SetPos(65, 18);
	printf("不能穿墙,不能碰到自己,");
	SetPos(65, 19);
	printf("按 ↑、↓、←、→ 控制蛇蛇的移动,");
	SetPos(65, 21);
	printf("F1为加速,F2为减速,");
	SetPos(65, 22);
	printf("ESC:退出游戏 SPACE:暂停");
}

//窗口相关设置
void ScreenPrepare()
{
	//设置窗体大小
	system("mode con cols=100 lines=40");
	system("title 贪吃蛇小游戏");
	//隐藏光标
	HideCurSor();
	//打印欢迎语
	Welcome();
	//打印地图
	DrawMap();
	//显示提示信息
	HelpInfo();
}

//设置颜色
void SetColor(int HeadColor)
{
	SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE), HeadColor);
}

//打印蛇
void PrintSnake(pSnake ps)
{
	pSnakeNode cur = ps->pSnake;
	int flag = 1;
	while (cur)
	{
		if (flag == 1)
		{
			//为蛇头设置颜色
			SetColor(12);
		}
		SetPos(cur->x, cur->y);//设置每个节点的位置
		wprintf(L"%lc", BODY);
		cur = cur->next;
		flag = 0;
		SetColor(10);
	}
}

//初始化蛇
void InitSnake(pSnake ps)
{
	int i = 0;
	pSnakeNode cur = NULL;
	//蛇的长度开始为3
	for (i = 0; i < 5; i++)
	{
		cur = (pSnakeNode)malloc(sizeof(SnakeNode));
		if (cur == NULL)
		{
			perror("InitSnake():malloc");
			return;
		}
		cur->next = NULL;
		cur->x = POS_X + i * 2;  //节点横坐标依次增加2
		cur->y = POS_Y;

		//头插法将节点相连
		if (ps->pSnake == NULL)
		{
			ps->pSnake = cur;
		}
		else
		{
			cur->next = ps->pSnake;
			ps->pSnake = cur;
		}
	}
	//打印蛇
	PrintSnake(ps);
	//初始化蛇的其它信息
	ps->Dire = RIGHT;
	ps->FoodScore = 10;
	ps->SleepTime = 200;
	ps->State = OK;
	ps->Score = 0;
}

//创建食物
void CreatFood(pSnake ps)
{
	int x = 0;
	int y = 0;
	//食物坐标应该在墙体内
again:
	do
	{
		x = rand() % 53 + 2;
		y = rand() % 25 + 1;
	} while (x % 2 != 0);

	//食物坐标不与蛇重叠
	pSnakeNode cur = ps->pSnake;
	while (cur)
	{
		if (x == cur->x && y == cur->y)
		{
			goto again;
		}
		cur = cur->next;
	}
	//根据坐标创建食物
	pSnakeNode pFood = (pSnakeNode)malloc(sizeof(SnakeNode));
	if (pFood == NULL)
	{
		perror("CreatFood():malloc()");
		return;
	}
	//打印食物
	pFood->x = x;
	pFood->y = y;
	SetColor(14);
	SetPos(pFood->x, pFood->y);
	wprintf(L"%lc", FOOD);
	SetColor(10);
	ps->pFood = pFood;
}

//游戏准备
void GameStart(pSnake psnake)
{
	//设置好窗口、地图
	ScreenPrepare();
	//初始化蛇
	InitSnake(psnake);
	//创建食物
	CreatFood(psnake);
}

//暂停
void pause()
{
	while (1)
	{
		//一直休眠,直到再次按空格键
		Sleep(200);
		if (KEY_PRESS(VK_SPACE))
		{
			break;
		}
	}
}

//玩游戏
void GameRun(pSnake ps)
{
	do
	{
		//显示分数
		SetColor(12);
		SetPos(65, 12);
		printf("目前得分:%-5d",ps->Score);
		SetColor(10);
		SetPos(65, 13);
		printf("每个食物:%2d分",ps->FoodScore);
		//检测按键
		if (KEY_PRESS(VK_UP) && ps->Dire != DOWN)
		{
			//按上键,且蛇的方向不能向下
			ps->Dire = UP;
		}
		else if (KEY_PRESS(VK_DOWN) && ps->Dire != UP)
		{
			//按下键,且蛇的方向不能向上
			ps->Dire = DOWN;
		}
		else if (KEY_PRESS(VK_LEFT) && ps->Dire != RIGHT)
		{
			//按左键,且蛇的方向不能向右
			ps->Dire = LEFT;
		}
		else if (KEY_PRESS(VK_RIGHT) && ps->Dire != LEFT)
		{
			//按右键,且蛇的方向不能向左
			ps->Dire = RIGHT;
		}
		else if (KEY_PRESS(VK_ESCAPE))
		{
			//按ESC,退出游戏
			ps->State = END;
		}
		else if (KEY_PRESS(VK_SPACE))
		{
			//按空格,暂停
			pause();
		}
		else if (KEY_PRESS(VK_F1))
		{
			//按F1加速,即睡眠时间变短
			
			//休眠时间不能是负数,最快就是休眠30ms
			if (ps->SleepTime >= 50)
			{
				ps->SleepTime -= 20;
				//速度变快,食物的分数变高
				ps->FoodScore += 2;
			}
		}
		else if (KEY_PRESS(VK_F2))
		{
			//F2减速,睡眠时间变长

			//食物的分数不能减到负数,最多减为1分
			if (ps->FoodScore >= 3)
			{
				ps->SleepTime += 20;
				ps->FoodScore -= 2;
			}
		}
		//按照蛇的睡眠时间,真正实现休眠
		Sleep(ps->SleepTime);

		//休眠后,蛇要移动
		SnakeMove(ps);

	} while (ps->State == OK);//只有蛇的状态为OK,才能继续检测按键
}

//移动蛇
void SnakeMove(pSnake ps)
{
	//先产生节点
	pSnakeNode pNext = (pSnakeNode)malloc(sizeof(SnakeNode));
	if (pNext == NULL)
	{
		perror("SnakeMove():malloc()");
		return;
	}
	//根据蛇的方向,设定节点的位置
	switch (ps->Dire)
	{
	case UP:
		//蛇的方向向上,节点应该在当前蛇头的上方;横坐标不变,纵坐标减1
		pNext->x = ps->pSnake->x;
		pNext->y = ps->pSnake->y - 1;
		break;
	case DOWN:
		//蛇的方向向下,节点应该在当前蛇头的下方;横坐标不变,纵坐标加1
		pNext->x = ps->pSnake->x;
		pNext->y = ps->pSnake->y + 1;
		break;
	case LEFT:
		//蛇的方向向左,节点应该在当前蛇头的左边;纵坐标不变,横坐标减2
		pNext->x = ps->pSnake->x - 2;
		pNext->y = ps->pSnake->y;
		break;
	case RIGHT:
		//蛇的方向向右,节点应该在当前蛇头的右方;纵坐标不变,横坐标加2
		pNext->x = ps->pSnake->x + 2;
		pNext->y = ps->pSnake->y;
		break;
	}

	//判断下一个节点是否是食物
	if (JudgeNext(ps, pNext))
	{
		//是食物,吃掉,长度增加
		EatFood(ps, pNext);
	}
	else
	{
		//不是食物,吃掉,长度不增加
		NoFood(ps, pNext);
	}

	//撞墙
	KillByWall(ps);

	//撞自己
	KillBySelf(ps);
}

//判断节点是否是食物
int JudgeNext(pSnake ps, pSnakeNode pNext)
{
	return ((ps->pFood->x == pNext->x) && (ps->pFood->y == pNext->y));
}

//是食物,吃掉,长度增加
void EatFood(pSnake ps, pSnakeNode pNext)
{
	//吃掉食物,头插法将节点插入
	pNext->next = ps->pSnake;
	ps->pSnake = pNext;
	//打印蛇
	PrintSnake(ps);
	//加分
	ps->Score +=  ps->FoodScore;
	//释放食物节点
	free(ps->pFood);
	//再次创建食物
	CreatFood(ps);
}

//不是食物
void NoFood(pSnake ps, pSnakeNode pNext)
{
	//头插法连接
	pNext->next = ps->pSnake;
	ps->pSnake = pNext;
	//删除蛇尾
	pSnakeNode cur = ps->pSnake;
	//找到蛇尾的前一个节点
	while (cur->next->next)
	{
		cur = cur->next;
	}
	pSnakeNode del = cur->next;
	//将蛇尾打印为空格
	SetPos(del->x, del->y);
	printf("  ");
	//释放蛇尾节点
	free(del);
	cur->next = NULL;
	//打印蛇
	PrintSnake(ps);
}

//撞墙
void KillByWall(pSnake ps)
{
	if (ps->pSnake->x == 0 ||
		ps->pSnake->x == 56 ||
		ps->pSnake->y == 0 ||
		ps->pSnake->y == 26)
	{
		ps->State = 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->State = KILL_BY_SELF;
		}
		cur = cur->next;
	}
}

//游戏结束
void GameEnd(pSnake ps)
{
	SetPos(20, 13);
	SetColor(12);

	switch (ps->State)
	{
	case END:
		printf("您结束了游戏");
		break;
	case KILL_BY_SELF:
		printf("很遗憾!您撞到了自己");
		break;
	case KILL_BY_WALL:
		printf("很遗憾!您撞墙了");
		break;
	}
	pSnakeNode cur = ps->pSnake;
	while (cur)
	{
		pSnakeNode del = cur;
		cur = cur->next;
		free(del);
	}
	ps = NULL;
	SetColor(10);

}

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

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

相关文章

TikTok直播对网络环境的要求是怎么样的

TikTok直播作为一种互动性强、实时性要求高的社交媒体形式&#xff0c;对网络环境有着一系列特定的需求。了解并满足这些需求&#xff0c;对于确保用户体验、提高直播质量至关重要。本文将深入探讨TikTok直播对网络环境的要求以及如何优化网络设置以满足这些要求。 TikTok直播的…

Django学习之小试牛刀

六、Django学习之小试牛刀 其他关于Python Web开发笔记&#xff1a;&#xff08;如果遇到问题可以一起交流~&#xff09; 一、Flask学习之HTML-CSDN博客 二、Flask学习之CSS-CSDN博客 【接上篇】二、Flask学习之CSS&#xff08;下篇&#xff09;-CSDN博客 三、Flask学习之B…

读取一个batch的图像并且显示出来

1读取一个batch用于训练 我们在训练模型的时候&#xff0c;除了观察图像的标签和尺寸&#xff0c;最好能读取一个batch的图像显示出来&#xff0c;观察原始图像和grountruth是否对应&#xff0c;如果正确才能正式开始后续的训练。 下面以一个皮肤病分割的数据集加以演示。 2…

漏洞原理SQL注入 手工注入漏洞

漏洞原理SQL注入 手工注入漏洞 SQL 注入的前置知识 information_schema库information_schema 是mysql5.0以上版本中自带的一个数据库。tables表information_schema库中的tables表中table_schema列&#xff08;存储数据库名&#xff09;和table_name列&#xff08;存储表名&…

【第六天】蓝桥杯备战

题 1、明明的随机数2、特殊日期 1、明明的随机数 https://www.lanqiao.cn/problems/539/learning/ 解法&#xff1a;暴力 import java.util.Scanner; import java.util.Arrays; // 1:无需package // 2: 类名必须Main, 不可修改public class Main {public static void main…

node学习过程中的终端命令

冷的哥们手真tm冷&#xff0c;打字都是僵的&#xff0c;屮 目录 一、在学习nodejs过程中用到的终端命令总结 一、在学习nodejs过程中用到的终端命令 node -v nvm install 20.11.0 nvm list nvm list available nvm on nvm -v nvm use 20.11.0 node加要运行的js文件路径 ps&a…

手指伸不直,锤状指不可忽视!

在日常生活和工作中&#xff0c;手指很容易戳伤&#xff0c;损伤后有时出现末节手指伸不直&#xff0c;影响手指屈伸活动障碍而就诊&#xff0c;医生会说手指损伤形成锤状指。那么什么是锤状指&#xff1f; 01 什么是“锤状指畸形”&#xff1f; 锤状指是指伸肌腱止点断裂后的…

确知信号的类型:能量信号与功率信号

通信原理第17页第一段&#xff1a; 例如&#xff0c; s ( t ) 8 s i n ( 5 t 1 ) , − ∞ < t < ∞ s(t)8sin(5t1),-\infty<t<\infty s(t)8sin(5t1),−∞<t<∞&#xff0c;就属于周期信号&#xff0c;其周期 T 0 2 π / 5 T_02\pi/5 T0​2π/5 三角函数很…

Spring5深入浅出篇:Spring中ioc(控制反转)与DI(依赖注入)

Spring5深入浅出篇:Spring中ioc(控制反转)与DI(依赖注入) 反转(转移)控制(IOC Inverse of Control) 控制&#xff1a;对于成员变量赋值的控制权 反转控制&#xff1a;把对于成员变量赋值的控制权&#xff0c;从代码中反转(转移)到Spring⼯⼚和配置⽂件中完成好处&#xff1a;…

docker 部署xxl-job

docker 部署xxl-job XXL-JOB github地址 https://github.com/xuxueli/xxl-job XXL-JOB 文档地址 https://www.xuxueli.com/xxl-job/ XXL-JOB是一个分布式任务调度平台&#xff0c;其核心设计目标是开发迅速、学习简单、轻量级、易扩展。现已开放源代码并接入多家公司线上产品…

Ubuntu apt update提示:GPG 缺少公钥解决方法

Ubuntu 运行: sudo apt update #or sudo apt-get update提示&#xff1a;GPG 缺少公钥以及404 Not Found&#xff0c;如下面所示&#xff0c;有mirror.bwbot.org 和ppa.launchpadcontent.net两个源出现问题。 好多网友用后面的方法解决 真正解决&#xff1a;gpg --verify sig:…

数据库ER图相关概念及其画法

ER图基本概念 ER图&#xff08;Entity-Relationship Diagram&#xff09;是一种用于描述现实世界概念模型的图形化表示方法&#xff0c;通过使用图形符号和元素来表示实体、属性和它们之间的关系。在ER图中&#xff0c;实体、属性和关系分别使用不同的图形元素来表示&#xff0…

【前端web入门第二天】01 html语法实现列表与表格_合并单元格

html语法实现列表与表格 文章目录: 1.列表 1.1 无序列表1.2 有序列表1.3 定义列表 2.表格 2.1 表格基本结构2.2 表格结构标签2.3 合并单元格 写在最前,第二天学习目标: 列表 表格 表单 元素为嵌套关系 1.列表 作用:布局内容排列整齐的区域。 列表分类:无序列表、有序列表…

Redis中BigKey的分析与优化

Redis中BigKey的分析与优化 Redis以其出色的性能和易用性&#xff0c;在互联网技术栈中占据了重要的地位。 但是&#xff0c;高效的工具使用不当也会成为性能瓶颈。在Redis中&#xff0c;BigKey是常见的性能杀手之一&#xff0c;它们会消耗过多的内存&#xff0c;导致网络拥塞…

专有钉钉开发记录,及问题总结

先放几个专有钉钉开发文档 专有钉钉官网的开发指南 服务端(后端)api文档 前端api文档 前端开发工具下载地址 小程序配置文件下载地址 后端SDK包下载地址 专有钉钉域名是openplatform.dg-work.cn 开发记录 开发专有钉钉时有时会遇到要使用钉钉的api&#xff1b;通过 my 的方…

【leetcode题解C++】101.对称二叉树 and 111.二叉树的最小深度 and 222.完全二叉树的节点个数 and 110.平衡二叉树

101. 对称二叉树 给你一个二叉树的根节点 root &#xff0c; 检查它是否轴对称。 示例 1&#xff1a; 输入&#xff1a;root [1,2,2,3,4,4,3] 输出&#xff1a;true示例 2&#xff1a; 输入&#xff1a;root [1,2,2,null,3,null,3] 输出&#xff1a;false 思路&#xff1a…

Pandas.Series.mode() 众数 详解 含代码 含测试数据集 随Pandas版本持续更新

关于Pandas版本&#xff1a; 本文基于 pandas2.2.0 编写。 关于本文内容更新&#xff1a; 随着pandas的stable版本更迭&#xff0c;本文持续更新&#xff0c;不断完善补充。 传送门&#xff1a; Pandas API参考目录 传送门&#xff1a; Pandas 版本更新及新特性 传送门&…

Blender教程(基础)-物体添加-03

1、打开的界面如下图会存在3个物体、英文状态下按键盘字母A全选、然后按键盘delete删除。 删除后一片空白 2、新增物体 方式1&#xff1a;在英文状态下按键盘shiftA组合键弹出如下添加物体弹窗 方式2&#xff1a;在菜单下找到添加点击弹出添加选项 3、举例新增物体 采用上述…

MongoDB本地部署并结合内网穿透实现公网访问本地数据库

文章目录 前言1. 安装数据库2. 内网穿透2.1 安装cpolar内网穿透2.2 创建隧道映射2.3 测试随机公网地址远程连接 3. 配置固定TCP端口地址3.1 保留一个固定的公网TCP端口地址3.2 配置固定公网TCP端口地址3.3 测试固定地址公网远程访问 4. 结语 前言 MongoDB是一个基于分布式文件…

Gin 应用多实例部署session问题、session参数与刷新

文章目录 一、Gin Session 存储的实现方案二、memstore&#xff1a;基于内存的实现2.1 基本使用2.2 关键参数 三、使用redis&#xff1a;多实例部署3.1 使用redis优势3.2 基本使用 四、信息安全的三个核心概念五、Gin Session 参数5.1 参数介绍 六、Session 自动刷新 一、Gin S…