贪吃蛇(c实现)

news2024/11/28 4:45:00

目录

游戏说明:

 第一个是又是封面,第二个为提示信息,第三个是游戏运行界面

游戏效果展示:

 游戏代码展示:

snack.c

 test.c

 snack.h

控制台程序的准备:

控制台程序名字修改:

 参考:mode命令(mode | Microsoft Learn)

游戏框架构建:

控制台屏幕上的坐标COORD:

隐藏光标:

 光标跳转

打印颜色设置:

初始化界面:

需要注意的地方就是:

 例如第一次的坐标就为 (i,j) 那么下一次坐标就为(i+2,j);

 宽字符打印准备:

初始化蛇与蛇的打印:

随机创建食物:

蛇的单向移动:

大致小部分已经实现完成,那么就利用游戏逻辑来实现剩余的代码;

添加方向的改变与判断蛇的各个状态判断:

 对该函数里面各个小函数进行代码展示:

最后,还有对应的就是运行是代码的逻辑展示

速度的控制(单位毫秒):

 

 最后的游戏收尾就是提示信息的打印:

最后一步便是锦上添花了,就是打印提示信息:

到这里就已经完成了,一共有三个页面:

 第一个是又是封面,第二个为提示信息,第三个是游戏运行界面


 

游戏说明:

  1. 按方向键上下左右,可以实现蛇移动方向的改变
  2. 按F3加速,F4减速
  3. 按ESC正常退出游戏,按空格暂停游戏
  4. 加速可以获得更多的分数
  5. 获得100即可获得胜利

 (待优化部分:背景音乐,记录历史最高得分)

 第一个是又是封面,第二个为提示信息,第三个是游戏运行界面

游戏效果展示:

贪吃蛇游戏当中蛇的移动速度可以进行调整,动图当中把速度调得较慢(速度太快导致动图上蛇身显示不全),下面给出的代码当中将蛇的速度调整到了合适的位置,大家可以试试。

贪吃蛇

 

 游戏代码展示:

snack.c

#define  _CRT_SECURE_NO_WARNINGS

#include"snack.h"
void color(int c)
{
	SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE), c); //颜色设置
	//注:SetConsoleTextAttribute是一个API(应用程序编程接口)
}
void cursor_hide()
{
	HANDLE hOutput = GetStdHandle(STD_OUTPUT_HANDLE);
	CONSOLE_CURSOR_INFO CursorInfo;
	GetConsoleCursorInfo(hOutput, &CursorInfo);//获取控制台光标信息
	CursorInfo.bVisible = false; //隐藏控制台光标 
	SetConsoleCursorInfo(hOutput, &CursorInfo);//设置控制台光标状态 
}

void SetPos(int x,int y)
{
	//获得标准输出设备的句柄
	HANDLE hOutput = NULL;
	hOutput = GetStdHandle(STD_OUTPUT_HANDLE);

	//定位光标的位置,到pos
	COORD pos = { x,y };
	SetConsoleCursorPosition(hOutput, pos);
}

void Welcome_game()
{
	color(9);
	SetPos(35, 12);
	wprintf(L"欢迎来到贪吃蛇小游戏\n");
	SetPos(36, 18);
	system("pause");
	system("cls");
	SetPos(25, 14);
	wprintf(L"用 ↑.↓.←.→ 来控制蛇的移动");
	SetPos(25, 15);
	wprintf(L"按F3加速,F4减速");
	SetPos(25, 16);
	wprintf(L"按ESC正常退出游戏,按空格暂停游戏");
	SetPos(25, 17);
	wprintf(L"加速可以获得更多的分数");
	SetPos(25, 18);
	wprintf(L"由能力有限公司提供技术支持");
	SetPos(0, 25);
	system("pause");
	system("cls");
	color(7);
}

void CreatMap()
{
	color(12);
	//上
	for (int i = 0; i < 29; i++)
	{
		wprintf(L"%lc", L'□');
	}
	//下
	SetPos(0, 26);
	for (int i = 0; i < 29; i++)
	{
		wprintf(L"%lc", L'□');
	} 
	//左
	for (int i = 1; i <= 25; i++)
	{
		SetPos(0, i);
		wprintf(L"%lc", L'□');
	}
	//右
	for (int i = 1; i <= 25; i++)
	{
		SetPos(56, i);
		wprintf(L"%lc", L'□');
	}
	color(7);
}



void CreateSnack(pSnack ps)
{
	color(10);
	//默认开始初始化为5个结点
	pSnackNode cur = NULL;
	for (int i = 0; i < 5; i++)
	{
		cur =(pSnackNode)malloc(sizeof(SnackNode));
		cur->next = NULL;
		cur->x = POS_X + i * 2;
		cur->y = POS_Y;
		if (ps->_psnack == NULL)//第一次头插
		{
			ps->_psnack = cur;
		}
		else
		{
			cur->next = ps->_psnack;
			ps->_psnack = cur;
		}

	}
	while (cur)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%lc", Body);
		cur = cur->next;
	}
	cur= ps->_psnack;

	//设置蛇的相关信息
	ps->_dir = RIGHT;
	ps->_food_weight = 10;
	ps->_sleep_time = 200;
	ps->_status = OK;
	color(7);
}

void CreateFood(pSnack ps)
{
	int x = 0;//2-54
	int y = 0;//1-25
	again:
	do{
		x = rand() % 53 + 2;
		y = rand() % 24 + 1;
	} while (x%2!=0);

	//检测改坐标是否与蛇身重合 
	pSnackNode cur = ps->_psnack;
	while (cur)
	{
		if (cur->x == x && cur->y == y)
		{
			goto again;
		}
		cur = cur->next;
	}
	pSnackNode SnackFood = (pSnackNode)malloc(sizeof(SnackNode));
	if (SnackFood == NULL)
	{
		perror("CreateFood malloc fail");
		return;
	}
	color(13);
	SnackFood->x = x;
	SnackFood->y = y;
	SnackFood->next = NULL;
	SetPos(x, y);
	wprintf(L"%lc", Food);
	ps->_pfood = SnackFood;
	color(7);
}

void Pause()
{
	while (1)
	{
		Sleep(200);
		if (KEY_PRESS(VK_SPACE))
		{
			break;
		}
	}
}

void GameStart(pSnack ps)
{
	system("mode con cols=100 lines=30");
	system("title 贪吃蛇");
	//1光标隐藏
	cursor_hide();

	//2.打印环境界面
	//第一个界面,欢迎
	// 第二个界面,介绍怎么操作游戏
	Welcome_game();//+3.功能介绍
	CreatMap();
	

	//创建蛇
	CreateSnack(ps);
	//创建食物
	CreateFood(ps);


	//SetPos(0, 29);
	//system("pause");
}

void PrintHelpInfo()
{
	color(15);
	SetPos(64, 10);
	wprintf(L"不能穿墙,不能咬到自己");
	SetPos(64, 12);
	wprintf(L"用 ↑.↓.←.→ 来控制蛇的移动");
	SetPos(64, 14);
	wprintf(L"按F3加速,F4减速");
	SetPos(64, 16);
	wprintf(L"按ESC正常退出游戏,按空格暂停游戏");
	SetPos(64, 18);
	wprintf(L"能力有限公司提供支持");
	SetPos(0, 29);
	//	system("pause");
	color(7);
}

bool Next_Is_Food(pSnackNode pn, pSnack ps)
{
	return (ps->_pfood->x == pn->x && ps->_pfood->y == pn->y);
}

void Eat_Food(pSnackNode pn, pSnack ps)
{
	ps->_pfood->next = ps->_psnack;
	ps->_psnack = ps->_pfood;
	free(pn);
	pn = NULL;
	pSnackNode cur = ps->_psnack;
	color(10);
	while (cur)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%lc", Body);
		cur = cur->next;
	}
	color(7);
	ps->_sum_score += ps->_food_weight;
	//在重新生成食物
	CreateFood(ps);
	
}

void No_Food(pSnackNode pn, pSnack ps)
{
	pn->next = ps->_psnack;
	ps->_psnack = pn;
	pSnackNode cur = ps->_psnack;
	//打印出来五个了
	color(10);
	while (cur->next->next != NULL)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%lc", Body);
		cur = cur->next;
	}
	color(7);
	//将第六个位置打印为空格
    //释放第六个
	SetPos(cur->next->x, cur->next->y);
	printf("  ");
	
	free(cur->next);
	//再将倒数第二个的next为NULL
	cur->next = NULL;
}

void Kill_By_Wall(pSnack ps)
{
	if(ps->_psnack->x == 0 || ps->_psnack->x == 56 ||
		ps->_psnack->y == 0 || ps->_psnack->y == 26)
	{
		ps->_status = KILL_BY_WALL;
	}
}

void Kill_By_Self(pSnack ps)
{
	pSnackNode cur = ps->_psnack -> next;
	while (cur)
	{
		if (cur->x == ps->_psnack->x && cur->y == ps->_psnack->y)
		{
			ps->_status = KILL_BY_SELF;
			break;
		}
		cur = cur->next;
	}
}

void SnackMove(pSnack ps)
{
	pSnackNode pNextNode = (pSnackNode)malloc(sizeof(SnackNode));
	if (pNextNode == NULL)
	{
		perror("SnackMove::malloc fail");
		return;
	}
	switch (ps->_dir)
	{
	case UP:
		pNextNode->x = ps->_psnack->x;
		pNextNode->y = ps->_psnack->y - 1;
		break;
	case DOWN:
		pNextNode->x = ps->_psnack->x;
		pNextNode->y = ps->_psnack->y + 1;
		break;
	case LEFT:
		pNextNode->x = ps->_psnack->x - 2;
		pNextNode->y = ps->_psnack->y;
		break;
	case RIGHT:
		pNextNode->x = ps->_psnack->x + 2;
		pNextNode->y = ps->_psnack->y;
		break;
	}
	//下一个位置是食物
	if (Next_Is_Food(pNextNode, ps))
	{
		Eat_Food(pNextNode,ps);
	}
	else
	{
		No_Food(pNextNode, ps);
	}

	//检测是否被撞墙死
	Kill_By_Wall(ps);

	//检测是否被撞自己死
	Kill_By_Self(ps);
}

void GameRun(pSnack ps)
{
	PrintHelpInfo();
	//SetPos(64, 10);
	do {
		SetPos(64, 6);
		wprintf(L"当前的总分数为:");
		printf("%d ", ps->_sum_score);
		SetPos(64, 8);
		wprintf(L"当前单个食物分数为:");
		printf("%d ", ps->_food_weight);

		if (KEY_PRESS(VK_UP) && ps->_dir != DOWN)
		{
			ps->_dir = UP;
		}
		else if (KEY_PRESS(VK_DOWN) && ps->_dir != UP)
		{
			ps->_dir = DOWN;
		}
		else if (KEY_PRESS(VK_LEFT) && ps->_dir != RIGHT)
		{
			ps->_dir = LEFT;
		}
		else if (KEY_PRESS(VK_RIGHT) && ps->_dir != LEFT)
		{
			ps->_dir = RIGHT;
		}
		//暂停  退出  加速  减速  
		else if (KEY_PRESS(VK_SPACE))
		{
			Pause();
		}
		else if (KEY_PRESS(VK_ESCAPE))
		{
			//正常退出游戏
			ps->_status = END_OK;
		}
		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;
			}
		}

		//实现蛇的移动
		SnackMove(ps);
		Sleep(ps->_sleep_time);
	} while (ps->_status==OK);
	//移动
	//实施打印情况
}



void GameEnd(pSnack ps)
{
	SetPos(24, 12);
	switch (ps->_status)
	{
	case END_OK:
		wprintf(L"您主动结束游戏\n");
		break;
	case KILL_BY_WALL:
		wprintf(L"您撞到墙上,游戏结束\n");
		break;
	case KILL_BY_SELF:
		wprintf(L"您撞到了自己,游戏结束\n");
		break;
	}

	//释放蛇身的链表

	pSnackNode cur = ps->_psnack;
	while (cur)
	{
		pSnackNode del = cur;
		cur = cur->next;
		free(del);
	}
}

 test.c

#define  _CRT_SECURE_NO_WARNINGS

#include"snack.h"


void test()
{
	char ch;
	do {
		//创建贪吃蛇
		Snack snack = { 0 };
		GameStart(&snack);
		//运行游戏
		GameRun(&snack);
		//结束游戏 - 善后工作
		GameEnd(&snack);
		SetPos(20, 15);
		printf("再来一局吗?(Y/N):");
		ch = getchar();
		while (getchar() != '\n');
	} while (ch == 'Y' || ch == 'y');
	SetPos(0, 28);

}
int main()
{
	srand((unsigned int)time(NULL));
	setlocale(LC_ALL, "");

	test();
	return 0;
}

 snack.h

#define  _CRT_SECURE_NO_WARNINGS

#include<stdio.h>
#include<windows.h>
#include<stdbool.h>
#include<stdlib.h>
#include <locale.h>
#include<time.h>
#include<errno.h>
#include<assert.h>

#define Wall L'□'
#define Body L'●'
#define Food L'★'

#define POS_X 24
#define POS_Y 5

//检查某个按键是否被按了
#define KEY_PRESS(VK) ( (GetAsyncKeyState(VK) & 0x1) ? 1 : 0 )



//蛇的方向
enum DIRECTION
{
	UP = 1,
	DOWN,
	LEFT,
	RIGHT,
};

//蛇的状态
enum GAME_STATUS
{
	OK,//正常
	KILL_BY_WALL,
	KILL_BY_SELF,
	END_OK,
};

typedef struct SnackNode
{
	int x;
	int y;
	//指向下一个结点
	struct SnackNode* next;
}SnackNode, * pSnackNode;

typedef struct Snack
{
	//蛇的头
	pSnackNode _psnack;
	pSnackNode _pfood;
	enum DIRECTION _dir;
	enum GAME_STATUS _status;
	int _food_weight;//一个食物的分数
	int _sum_score;//总成绩
	int _sleep_time;//蛇的速度,越小越快
}Snack,*pSnack;

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

//游戏开始
void GameStart(pSnack ps);

//欢迎界面
void Welcome_game();

//绘制地图
void CreatMap();

//打印提示操作信息
void PrintHelpInfo();

//创建蛇
void CreateSnack(pSnack ps);

//创建食物
void CreateFood(pSnack ps);

//游戏暂停
void Pause();

//游戏正常运行
void GameRun(pSnack ps);

//贪吃蛇的移动
void SnackMove(pSnack ps);

//检查下一个坐标位置是否为食物
bool Next_Is_Food(pSnackNode pn,pSnack ps);

//吃食物
void Eat_Food(pSnackNode pn, pSnack ps);

//下一个位置不是食物,进行移动
void No_Food(pSnackNode pn, pSnack ps);

//检测是否被撞墙死
void Kill_By_Wall(pSnack ps);

//检测是否被撞自己死
void Kill_By_Self(pSnack ps);

//正常的游戏结束
void GameEnd(pSnack ps);

 



控制台程序的准备:

需要运用到API

本游戏运行的时候需要用到控制台主机,而不是终端,对应的修改步骤如下:

控制台程序名字修改:

把名字改为贪吃蛇,会更好,那么修改方式如下:

 参考:mode命令(mode | Microsoft Learn)

system("title 贪吃蛇");

 

 

游戏框架构建:

首先定义游戏界面的大小,定义游戏区行数和列数。

平常我们运⾏起来的黑框程序其实就是控制台程序 我们可以使用cmd命令来设置控制台窗⼝的⻓宽:设置控制台窗⼝的大小,100行,30列

system("mode con cols=100 lines=30");

此外,我们还需要结构体用于表示蛇与食物的结点信息。

typedef struct SnackNode
{
	int x;
	int y;
	//指向下一个结点
	struct SnackNode* next;
}SnackNode, * pSnackNode;

此外还有存放游戏蛇的信息与各个游戏相关信息,也需要用结构体封装起来存放:

typedef struct Snack
{
	//蛇的头
	pSnackNode _psnack;
	pSnackNode _pfood;
	enum DIRECTION _dir;
	enum GAME_STATUS _status;
	int _food_weight;//一个食物的分数
	int _sum_score;//总成绩
	int _sleep_time;//蛇的速度,越小越快
}Snack,*pSnack;

同样也需要存放蛇的状态,比如正常,撞墙死亡,撞自己死亡。

//蛇的状态
enum GAME_STATUS
{
	OK,//正常
	KILL_BY_WALL,
	KILL_BY_SELF,
	END_OK,
};

同样蛇的运行方向也需要用一个枚举来存放:

为了增加可读性,我们使用一个数字来定义方向,如向上为1;

//蛇的方向
enum DIRECTION
{
	UP = 1,
	DOWN,
	LEFT,
	RIGHT,
};


控制台屏幕上的坐标COORD:

COORD是WindowsAPI中定义的⼀个结构体,表⽰⼀个字符在控制台屏幕幕缓冲区上的坐标,坐标系 (0,0)的原点位于缓冲区的顶部左侧单元格。

隐藏光标:

隐藏光标比较简单,是运用到WIN 32 API,先通过etConsoleCursorInfo(hOutput, &CursorInfo);获取控制台光标信息,再隐藏控制台光标,设置控制台光标状态;

void cursor_hide()
{
	HANDLE hOutput = GetStdHandle(STD_OUTPUT_HANDLE);
	CONSOLE_CURSOR_INFO CursorInfo;
	GetConsoleCursorInfo(hOutput, &CursorInfo);//获取控制台光标信息
	CursorInfo.bVisible = false; //隐藏控制台光标 
	SetConsoleCursorInfo(hOutput, &CursorInfo);//设置控制台光标状态 
}

 光标跳转

光标跳转,也就是让光标跳转到获得标准输出设备的句柄,与隐藏光标的操作步骤类似,然后定位光标的位置,跳转到指定位置:

void SetPos(int x,int y)
{
	//获得标准输出设备的句柄
	HANDLE hOutput = NULL;
	hOutput = GetStdHandle(STD_OUTPUT_HANDLE);

	//定位光标的位置,到pos
	COORD pos = { x,y };
	SetConsoleCursorPosition(hOutput, pos);
}

打印颜色设置:

 颜色设置函数的作用是,将此后输出的内容颜色都更为所指定的颜色,接收的参数c是颜色代码,十进制颜色代码表如下:

void color(int c)
{
	SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE), c); //颜色设置
	//注:SetConsoleTextAttribute是一个API(应用程序编程接口)
}

初始化界面:

第一步就为打印地图:

需要注意的地方就是:
  1. 在cmd窗口中一个小方块占两个单位的横坐标,一个单位的纵坐标。我们的墙使用宽字符进行对应的填充,
  2. 光标跳转函数SetPos接收的是光标将要跳至位置的横纵坐标。
 例如第一次的坐标就为 (i,j) 那么下一次坐标就为(i+2,j);

 宽字符打印准备:

1:需要引头文件:

#include <locale.h>

2:修改当前地区

	setlocale(LC_ALL, "");

 3:对应的字符

#define Wall L'□'
#define Body L'●'
#define Food L'★'

我设置的墙的颜色为红色,可以根据自己喜欢,自己根据上面的图给出的颜色进行调整。 

void CreatMap()
{
	color(12);
	//上
	for (int i = 0; i < 29; i++)
	{
		wprintf(L"%lc", L'□');
	}
	//下
	SetPos(0, 26);
	for (int i = 0; i < 29; i++)
	{
		wprintf(L"%lc", L'□');
	} 
	//左
	for (int i = 1; i <= 25; i++)
	{
		SetPos(0, i);
		wprintf(L"%lc", L'□');
	}
	//右
	for (int i = 1; i <= 25; i++)
	{
		SetPos(56, i);
		wprintf(L"%lc", L'□');
	}
	color(7);
}

初始化蛇与蛇的打印:

我默认开始蛇身加上蛇头一共五个结点大小:

最一开始蛇的坐标:

#define POS_X 24
#define POS_Y 5

我们的蛇是运用结构体,并运用的单链表来创造,那么我们打印只需要遍历就可以,还是比较简单的

初始化的代码如下:(蛇的颜色我设置的是绿色)

void CreateSnack(pSnack ps)
{
	color(10);
	//默认开始初始化为5个结点
	pSnackNode cur = NULL;
	for (int i = 0; i < 5; i++)
	{
		cur =(pSnackNode)malloc(sizeof(SnackNode));
		cur->next = NULL;
		cur->x = POS_X + i * 2;
		cur->y = POS_Y;
		if (ps->_psnack == NULL)//第一次头插
		{
			ps->_psnack = cur;
		}
		else
		{
			cur->next = ps->_psnack;
			ps->_psnack = cur;
		}

	}
	while (cur)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%lc", Body);
		cur = cur->next;
	}
	cur= ps->_psnack;

	//设置蛇的相关信息
	ps->_dir = RIGHT;
	ps->_food_weight = 10;
	ps->_sleep_time = 200;
	ps->_status = OK;
	color(7);
}

随机创建食物:

随机在游戏区生成食物,需要对生成后的坐标进行判断,只有该位置为空才能在此生成食物,否则需要重新生成坐标。食物坐标确定后,需要对游戏区该位置的状态进行标记。

食物我设置的是紫色。可以根据自己爱好,设置自己喜欢的颜色。

void CreateFood(pSnack ps)
{
	int x = 0;//2-54
	int y = 0;//1-25
	again:
	do{
		x = rand() % 53 + 2;
		y = rand() % 24 + 1;
	} while (x%2!=0);

	//检测改坐标是否与蛇身重合 
	pSnackNode cur = ps->_psnack;
	while (cur)
	{
		if (cur->x == x && cur->y == y)
		{
			goto again;
		}
		cur = cur->next;
	}
	pSnackNode SnackFood = (pSnackNode)malloc(sizeof(SnackNode));
	if (SnackFood == NULL)
	{
		perror("CreateFood malloc fail");
		return;
	}
	color(13);
	SnackFood->x = x;
	SnackFood->y = y;
	SnackFood->next = NULL;
	SetPos(x, y);
	wprintf(L"%lc", Food);
	ps->_pfood = SnackFood;
	color(7);
}

蛇的单向移动:

移动蛇函数的作用就是先覆盖当前所显示的蛇,然后再打印移动后的蛇。

对应蛇尾的位置打印变为空格并删除一次蛇尾,然后再次创建一个新的蛇头,更换蛇头

void SnackMove(pSnack ps)
{
	pSnackNode pNextNode = (pSnackNode)malloc(sizeof(SnackNode));
	if (pNextNode == NULL)
	{
		perror("SnackMove::malloc fail");
		return;
	}
	pNextNode->x = ps->_psnack->x + 2;
	pNextNode->y = ps->_psnack->y;

	pn->next = ps->_psnack;
	ps->_psnack = pn;
	pSnackNode cur = ps->_psnack;
	//打印出来五个了
	while (cur->next->next != NULL)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%lc", Body);
		cur = cur->next;
	}
	SetPos(cur->next->x, cur->next->y);
	printf("  ");
}


大致小部分已经实现完成,那么就利用游戏逻辑来实现剩余的代码;

在玩贪吃蛇的时候,我们知道

在蛇的移动过程,我们可以通过按键修改蛇的移动方向,来进行转弯,而且在移动的过程中,我们还可以随时改变速度,来改变游戏的难度,来增加游戏的可玩性,在行动的过程中,不免会撞墙,撞自己,吃到了食物,没有吃到食物,等等各种不同的情况,那么 对于实现的逻辑就是如上:

那么我先修改蛇的移动使其可以更换方向

在修改蛇的方向前,我们知道我们是通过按键来改变,那么我们就需要通过某种方法得知我们按了什么键来进行修改方向,同样也是API的知识

//检查某个按键是否被按了
#define KEY_PRESS(VK) ( (GetAsyncKeyState(VK) & 0x1) ? 1 : 0 )

添加方向的改变与判断蛇的各个状态判断:

bool Next_Is_Food(pSnackNode pn, pSnack ps)
{
	return (ps->_pfood->x == pn->x && ps->_pfood->y == pn->y);
}
void SnackMove(pSnack ps)
{
	pSnackNode pNextNode = (pSnackNode)malloc(sizeof(SnackNode));
	if (pNextNode == NULL)
	{
		perror("SnackMove::malloc fail");
		return;
	}
	switch (ps->_dir)
	{
	case UP:
		pNextNode->x = ps->_psnack->x;
		pNextNode->y = ps->_psnack->y - 1;
		break;
	case DOWN:
		pNextNode->x = ps->_psnack->x;
		pNextNode->y = ps->_psnack->y + 1;
		break;
	case LEFT:
		pNextNode->x = ps->_psnack->x - 2;
		pNextNode->y = ps->_psnack->y;
		break;
	case RIGHT:
		pNextNode->x = ps->_psnack->x + 2;
		pNextNode->y = ps->_psnack->y;
		break;
	}
	//下一个位置是食物
	if (Next_Is_Food(pNextNode, ps))
	{
		Eat_Food(pNextNode,ps);
	}
	else
	{
		No_Food(pNextNode, ps);
	}

	//检测是否被撞墙死
	Kill_By_Wall(ps);

	//检测是否被撞自己死
	Kill_By_Self(ps);
}

 对该函数里面各个小函数进行代码展示:

这里面的小函数都是比较好实现的,这里就不在解释:

Eat_Food(pNextNode,ps);

 需要注意的是吃完这个食物后,要记得重新随机创建食物

void Eat_Food(pSnackNode pn, pSnack ps)
{
	ps->_pfood->next = ps->_psnack;
	ps->_psnack = ps->_pfood;
	free(pn);
	pn = NULL;
	pSnackNode cur = ps->_psnack;
	color(10);
	while (cur)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%lc", Body);
		cur = cur->next;
	}
	color(7);
	ps->_sum_score += ps->_food_weight;
	//在重新生成食物
	CreateFood(ps);
}

void No_Food(pSnackNode pn, pSnack ps)

 要记得把尾打印改为空格

void No_Food(pSnackNode pn, pSnack ps)
{
	pn->next = ps->_psnack;
	ps->_psnack = pn;
	pSnackNode cur = ps->_psnack;
	//打印出来五个了
	color(10);
	while (cur->next->next != NULL)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%lc", Body);
		cur = cur->next;
	}
	color(7);
	//将第六个位置打印为空格
    //释放第六个
	SetPos(cur->next->x, cur->next->y);
	printf("  ");
	
	free(cur->next);
	//再将倒数第二个的next为NULL
	cur->next = NULL;
}

void Kill_By_Wall(pSnack ps)

void Kill_By_Wall(pSnack ps)
{
	if(ps->_psnack->x == 0 || ps->_psnack->x == 56 ||
		ps->_psnack->y == 0 || ps->_psnack->y == 26)
	{
		ps->_status = KILL_BY_WALL;
	}
}

void Kill_By_Self(pSnack ps)

void Kill_By_Self(pSnack ps)
{
	pSnackNode cur = ps->_psnack -> next;
	while (cur)
	{
		if (cur->x == ps->_psnack->x && cur->y == ps->_psnack->y)
		{
			ps->_status = KILL_BY_SELF;
			break;
		}
		cur = cur->next;
	}
}

最后,还有对应的就是运行是代码的逻辑展示

void GameRun(pSnack ps)
{

	//SetPos(64, 10);
	do {


		if (KEY_PRESS(VK_UP) && ps->_dir != DOWN)
		{
			ps->_dir = UP;
		}
		else if (KEY_PRESS(VK_DOWN) && ps->_dir != UP)
		{
			ps->_dir = DOWN;
		}
		else if (KEY_PRESS(VK_LEFT) && ps->_dir != RIGHT)
		{
			ps->_dir = LEFT;
		}
		else if (KEY_PRESS(VK_RIGHT) && ps->_dir != LEFT)
		{
			ps->_dir = RIGHT;
		}
		//暂停  退出  加速  减速  
		else if (KEY_PRESS(VK_SPACE))
		{
			Pause();
		}
		else if (KEY_PRESS(VK_ESCAPE))
		{
			//正常退出游戏
			ps->_status = END_OK;
		}
		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;
			}
		}

		//实现蛇的移动
		SnackMove(ps);
		Sleep(ps->_sleep_time);
	} while (ps->_status==OK);
	//移动
	//实施打印情况
}

速度的控制(单位毫秒):

 

void Pause()
{
	while (1)
	{
		Sleep(200);
		if (KEY_PRESS(VK_SPACE))
		{
			break;
		}
	}
}

 最后的游戏收尾就是提示信息的打印:

void GameEnd(pSnack ps)
{
	SetPos(24, 12);
	switch (ps->_status)
	{
	case END_OK:
		wprintf(L"您主动结束游戏\n");
		break;
	case KILL_BY_WALL:
		wprintf(L"您撞到墙上,游戏结束\n");
		break;
	case KILL_BY_SELF:
		wprintf(L"您撞到了自己,游戏结束\n");
		break;
	}

	//释放蛇身的链表

	pSnackNode cur = ps->_psnack;
	while (cur)
	{
		pSnackNode del = cur;
		cur = cur->next;
		free(del);
	}
}


最后一步便是锦上添花了,就是打印提示信息:

 

void PrintHelpInfo()
{
	color(15);
	SetPos(64, 10);
	wprintf(L"不能穿墙,不能咬到自己");
	SetPos(64, 12);
	wprintf(L"用 ↑.↓.←.→ 来控制蛇的移动");
	SetPos(64, 14);
	wprintf(L"按F3加速,F4减速");
	SetPos(64, 16);
	wprintf(L"按ESC正常退出游戏,按空格暂停游戏");
	SetPos(64, 18);
	wprintf(L"能力有限公司提供支持");
	SetPos(0, 29);
	//	system("pause");
	color(7);
}

到这里就已经完成了,一共有三个页面:

 第一个是又是封面,第二个为提示信息,第三个是游戏运行界面

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

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

相关文章

【考研数学】汤家凤“免单“数学题被吐槽‘太难’,老汤回应「怎么还有脸笑」,网友:这些题有毒!

我看了汤家凤老师出的几道题&#xff0c;实际上对于考研的同学来说&#xff0c;确实是送分题 第一个是三角函数变换中的万能公式&#xff1b;第二个e^x的泰勒展开公式&#xff1b;第三个是第一类重要极限。只要复习过&#xff0c;那基本上都能正常做出来。 至于汤家凤老师说「…

C#中数组与列表,集合等的联系

C#中&#xff0c;所有数组都自动继承于System.Array这个抽象类&#xff0c;数组都为引用类型&#xff0c; 所有对数组的更新都会导致源数组的元素值的篡改。 而所有集合的根都来自可枚举接口IEnumerable 数组有三种样式&#xff1a; 数组的Rank&#xff08;秩&#xff09;属…

纯血鸿蒙APP实战开发——Grid和List内拖拽交换子组件位置

Grid和List内拖拽交换子组件位置 介绍 本示例分别通过onItemDrop()和onDrop()回调&#xff0c;实现子组件在Grid和List中的子组件位置交换。 效果图预览 使用说明&#xff1a; 拖拽Grid中子组件&#xff0c;到目标Grid子组件位置&#xff0c;进行两者位置互换。拖拽List中子…

【数据结构】环状链表OJ题

✨✨✨专栏&#xff1a;数据结构 &#x1f9d1;‍&#x1f393;个人主页&#xff1a;SWsunlight 一、OJ 环形链表&#xff1a; 快慢指针即可解决问题: 2情况&#xff1a; 快指针走到结尾&#xff08;不是环&#xff09;快指针和尾指针相遇&#xff08;是环的&#xff09; …

macos使用yarn创建vite时出现Usage Error: The nearest package directory问题

步骤是macos上使用了yarn create vite在window上是直接可以使用了yarn但是在macos上就出现报错 我们仔细看&#xff0c;它说的If /Users/chentianyu isnt intended to be a project, remove any yarn.lock and/or package.json file there.说是要我们清除yarn.lock和package.js…

OGG几何内核-网格化的改进

OGG社区于4月19日发布了OGG 1.0 preview版本。相对于OCCT 7.7.0有很多改进&#xff0c;目前在持续研究中。最近测试了一下网格化&#xff0c;确实有很好的改进。对比展示如下&#xff1a; 几何内核&#xff1a; OGG 1.0 preview 几何内核&#xff1a;OCCT 7.7.0 采用OCCT几何内…

线程同步--条件变量,信号量

生产者和消费者模型 案例 /*生产者消费者模型&#xff08;粗略的版本&#xff09; */ #include <stdio.h> #include <pthread.h> #include <stdlib.h> #include <unistd.h>// 创建一个互斥量 pthread_mutex_t mutex;struct Node{int num;struct Node …

【ALM】ALM解决方案系列:质量保证

1软件的质量管理现状与痛点 在软件开发中&#xff0c;质量被视为软件产品的生命。保证软件质量&#xff0c;是贯穿整个软件生命周期的重要问题。如果在软件开发早期忽视质量管理&#xff0c;会导致软件项目管理出现问题。因此&#xff0c;重视并规范软件管理在软件项目管理中起…

[论文笔记]Corrective Retrieval Augmented Generation

引言 今天带来论文Corrective Retrieval Augmented Generation的笔记&#xff0c;这是一篇优化RAG的工作。 大型语言模型(LLMs) inevitable(不可避免)会出现幻觉&#xff0c;因为生成的文本的准确性不能仅仅由其参数化知识来确保。尽管检索增强生成(RAG)是LLMs的一个可行补充…

带你手撕红黑树! c++实现 带源码

目录 一、概念 二、特性 三、接口实现 1、插入 情况一&#xff1a;p为黑&#xff0c;结束 情况二&#xff1a;p为红 1&#xff09;叔叔存在且为红色 2&#xff09;u不存在/u存在且为黑色 &#xff08;1&#xff09;p在左&#xff0c;u在右 &#xff08;2&#xff09;…

【半夜学习MySQL】表的约束(含主键、唯一键、外键、zerofill、列描述、默认值、空属性详解)

&#x1f3e0;关于专栏&#xff1a;半夜学习MySQL专栏用于记录MySQL数据相关内容。 &#x1f3af;每天努力一点点&#xff0c;技术变化看得见 文章目录 前言空属性默认值列描述zerofill主键主键概述主键删除与追加复合主键 自增长唯一键外键综合案例 前言 上一篇文章中介绍了数…

Vue3项目打包部署到云服务器的Nginx中

文章目录 一、打包vue3项目二、dist文件夹上传到服务器三、改nginx配置文件Docker安装nginx 一、打包vue3项目 npm run build 二、dist文件夹上传到服务器 将dist文件夹放到docker安装的nginx中的html目录下 三、改nginx配置文件 然后重启nginx【改了配置文件重启nginx才能…

JVM的原理与性能

1 JVM 内存结构 1.1 运行时数据区 1.1.1 栈&#xff08;虚拟机栈&#xff09; 每个线程在创建时都会创建一个私有的Java虚拟机栈&#xff0c;在执行每个方法时都会打包成一个栈帧&#xff0c;存储了局部变量表、操作数栈、动态链接、方法出口等信息&#xff0c;然后放入栈中。…

关于我转生从零开始学C++这件事:获得神器

❀❀❀ 文章由不准备秃的大伟原创 ❀❀❀ ♪♪♪ 若有转载&#xff0c;请联系博主哦~ ♪♪♪ ❤❤❤ 致力学好编程的宝藏博主&#xff0c;代码兴国&#xff01;❤❤❤ 几天不见 &#xff0c;甚是想念&#xff01;哈咯大家好又是我大伟&#xff0c;五一的假期已经结束&#xff0…

rt-thread 挂载romfs与ramfs

参考&#xff1a; https://www.rt-thread.org/document/site/#/rt-thread-version/rt-thread-standard/tutorial/qemu-network/filesystems/filesystems?id%e4%bd%bf%e7%94%a8-romfs https://www.rt-thread.org/document/site/#/rt-thread-version/rt-thread-standard/tutor…

实操Linux磁盘管理(分区、格式化、挂载)

在Linux系统中&#xff0c;磁盘管理是一个必学的知识点。正确地进行磁盘分区、格式化和挂载可以确保我们能够充分利用磁盘空间并高效地存储和访问数据。 相比于Windows系统中的简单盘符管理&#xff0c;Linux中的磁盘管理更加复杂且灵活。在Linux系统中&#xff0c;一切设备都…

一款简易的免费抽奖软件

一、介绍 这款抽奖软件设计简洁&#xff0c;操作便捷。用户可以轻松将参与名单通过EXCEL文件导入至程序中&#xff0c;并可根据需要设定各类奖品和对应的中奖人数。在选定了奖品后&#xff0c;用户只需点击“开始”按钮&#xff0c;随后再按下“暂停”按钮&#xff0c;软件便会…

C++/Qt 小知识记录6

工作中遇到的一些小问题&#xff0c;总结的小知识记录&#xff1a;C/Qt 小知识6 dumpbin工具查看库导出符号OSGEarth使用编出的protobuf库&#xff0c;报错问题解决VS2022使用cpl模板后&#xff0c;提示会乱码的修改设置QProcess调用cmd.exe执行脚本QPainterPath对线段描边处理…

pycharm本地文件更新至虚拟机

tools–>deployment–>configuration root path的路径要跟远程路径对齐&#xff0c;方便后续运行 mapping映射&#xff0c;本地路径和远程路径 点击Browse 可以在右侧同步查看更新情况

【Element-UI快速入门】

文章目录 **Element-UI快速入门****一、Element-UI简介****二、安装Element-UI****三、引入Element-UI****四、使用Element-UI组件****五、自定义Element-UI组件样式****六、Element-UI布局组件****七、Element-UI表单组件****八、插槽&#xff08;Slots&#xff09;和主题定制…