贪吃蛇(c语言)!!源码加解析

目录

1.建议先把源码拿去VS中测试一下了解这个贪吃蛇是什么样的 

1.头文件代码

2.源代码

3.测试代码

4.代码详解

1.头文件的解析

2.源代码的解析 

 1.光标的位置封装函数

 2.打印欢迎界面

3.打印整体的一个地图 

4.蛇的初始化 (重要)

 5.打印边栏信息

6.设置用来暂停的函数 

7.  3个函数,关于走的下一步是不是食物 

 8.关于蛇是怎么死的

 9.蛇的移动(重点)

 10.游戏的运行(游戏的逻辑)

11.关于整体游戏的测试代码


1.建议先把源码拿去VS中测试一下了解这个贪吃蛇是什么样的 

1.头文件代码

#pragma once
#include <stdio.h>
#include <Windows.h>
#include <time.h>
#include<locale.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 5

typedef struct SnakeNode//蛇身
{
	int x;//蛇身的坐标
	int y;//蛇身的坐标
	struct SnakeNode* next;//下一个蛇身的节点
}SnakeNode ,* pSnakeNode;
typedef struct Snake//贪吃蛇整体
{
	pSnakeNode _pbody;//蛇身的指针
	pSnakeNode _pfood;//果实的指针
	enum DIRECTION _dir;//蛇的方向
	enum GAME_STATUS _status;//目前游戏的状态
	int _score;//当前的分数
	int _foodweight;//每个食物的得分
	int _sleeptime;//游戏的速度
}Snake,* pSnake;
enum DIRECTION//snake direction
{
	UP = 1,
	DOWN,
	LEFT,
	RIGHT
};
enum GAME_STATUS//the game.s status
{
	OK,
	KILL_BY_WALL,
	KILL_BY_SELF,
	END_NORMAL
};
//游戏的整体函数
void GameStart(pSnake ps);
void GameRun(pSnake ps);
void GameEnd(pSnake ps);
void SetPos(short x, short y);
void WelcomeToGame();
void PrintHelpInfo();//print some helpinfo
void CreatMap();
void InitSnake(pSnake ps);
void CreatFood(pSnake ps);
void pause();//game stop
int NextIsFood(pSnakeNode psn, pSnake ps);
void EatFood(pSnakeNode psn, pSnake ps);
void NotFood(pSnakeNode psn, pSnake ps);
int KillByWall(pSnake ps);
int KillBySelf(pSnake ps);
void SnakeMove(pSnake ps);

2.源代码


#include"tcs.h"
void SetPos(short x, short y)
{
	COORD pos = { x,y };
	HANDLE houtput = NULL;
	houtput = GetStdHandle(STD_OUTPUT_HANDLE);
	SetConsoleCursorPosition(houtput, pos);
}
void WelcomeToGame()
{
	SetPos(40, 15);
	printf("欢迎来到贪吃蛇\n");
	SetPos(40, 25);
	system("pause");
	system("cls");
	SetPos(25, 12);
	wprintf(L"⽤ ↑ . ↓ . ← . → 分别控制蛇的移动, F3为加速,F4为减速\n");
	SetPos(25, 13);
	printf("加速获得的分数更高\n");
	SetPos(40, 25);
	system("pause");
	system("cls");
}
void CreatMap()
{
	SetPos(0, 0);
	for (int i = 0; i < 58; i = i + 2)
	{
		wprintf(L"%c", WALL);
	}
	SetPos(0, 26);
	for (int i = 0; i < 58; i= i + 2)
	{
		wprintf(L"%c", WALL);
	}
	for (int i = 1; i < 26; i++)
	{
		SetPos(0, i);
		wprintf(L"%c", WALL);
	}
	for (int i = 1; i < 26; i++)
	{
		SetPos(56, i);
		wprintf(L"%c", WALL);
	}
}
void InitSnake(pSnake ps)
{
	pSnakeNode cur = NULL;
	for (int i = 0; i < 5; i++)
	{
		cur = (pSnakeNode)malloc(sizeof(SnakeNode));
		if (cur == NULL)
		{
			perror("snakenode don't get:");
			return;
		}
		cur->next = NULL;
		cur->x = POS_X + i * 2;
		cur->y = POS_Y;
		if (ps->_pbody == NULL)
		{
			ps->_pbody = cur;
		}
		else
		{
			cur->next = ps->_pbody;
			ps->_pbody = cur;
		}
	}

	//print snake
	cur = ps->_pbody;
	while (cur)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%c", BODY);
		cur = cur->next;
	}
	//init snake data
	ps->_dir = RIGHT;
	ps->_foodweight = 100;
	ps->_score = 0;
	ps->_sleeptime = 200;
	ps->_status = OK;
}
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->_pbody;
	while (cur)
	{
		if (cur->x == x && cur->y == y)
		{
			goto again;
		}
		cur = cur->next;
	}
	pSnakeNode pFood = (pSnakeNode)malloc(sizeof(SnakeNode));
	if (pFood == NULL)
	{
		perror("creat pfood failed,fool!\n");
		return;
	}
	else
	{
		pFood->x = x;
		pFood->y = y;
		SetPos(pFood->x, pFood->y);
		wprintf(L"%c", FOOD);
		ps->_pfood = pFood;
	}
}
void PrintHelpInfo()
{
	SetPos(64, 15);
	printf("不能创墙,也不能创自己");
	SetPos(64, 16);
	printf("用↑.↓.←.→分别控制蛇的移动.");
	SetPos(64, 17);
	printf("F3 为加速,F4 为减速\n");
	SetPos(64, 18);
	printf("ESC :退出游戏.space:暂停游戏.");
	SetPos(64, 20);
	printf("江子龙@版权");
}
void pause()
{
	while (1)
	{
		Sleep(300);
		if (KEY_PRESS(VK_SPACE))
		{
			break;
		}
	}
}
int NextIsFood(pSnakeNode psn, pSnake ps)
{
	return(psn->x == ps->_pfood->x) && (psn->y == ps->_pfood->y);
}
void EatFood(pSnakeNode psn, pSnake ps)
{
	psn->next = ps->_pbody;
	ps->_pbody = psn;
	pSnakeNode cur = ps->_pbody;
	while (cur)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%c", BODY);
		cur = cur->next;
	}
	ps->_score += ps->_foodweight;
	free(ps->_pfood);
	CreatFood(ps);
}
void NotFood(pSnakeNode psn, pSnake ps)
{
	psn->next = ps->_pbody;
	ps->_pbody = psn;
	pSnakeNode cur = ps->_pbody;
	while (cur->next->next)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%c", BODY);
		cur = cur->next;
	}
	SetPos(cur->next->x, cur->next->y);
	printf("  ");
	free(cur->next);
	cur->next = NULL;
}
int KillByWall(pSnake ps)
{
	if ((ps->_pbody->x == 0) || (ps->_pbody->x == 56) || (ps->_pbody->y == 0) || (ps->_pbody->y == 26))
	{
		ps->_status = KILL_BY_WALL;
		return 1;
	}
	return 0;
}
int KillBySelf(pSnake ps)
{
	pSnakeNode cur = ps->_pbody->next;
	while (cur)
	{
		if ((ps->_pbody->x == cur->x) && (ps->_pbody->y == cur->y))
		{
			ps->_status = KILL_BY_SELF;
			return 1;
		}
		cur = cur->next;
	}
	return 0;
}
void SnakeMove(pSnake ps)
{
	pSnakeNode pNextNode = (pSnakeNode)malloc(sizeof(SnakeNode));
	if (pNextNode == NULL)
	{
		perror("snake move open falied");
		return;
	}

	switch (ps->_dir)
	{
	case UP:
	{
		pNextNode->y = ps->_pbody->y - 1;
		pNextNode->x = ps->_pbody->x;
	}
		break;
	case DOWN:
	{
		pNextNode->y = ps->_pbody->y + 1;
		pNextNode->x = ps->_pbody->x;
	}
		break;
	case LEFT:
	{
		pNextNode->y = ps->_pbody->y ;
		pNextNode->x = ps->_pbody->x - 2;
	}
		break;
	case RIGHT:
	{
		pNextNode->y = ps->_pbody->y;
		pNextNode->x = ps->_pbody->x + 2;
	}
		break;
	default:
		break;
	}
	if (NextIsFood(pNextNode, ps))
	{
		EatFood(pNextNode, ps);
	}
	else
	{
		NotFood(pNextNode, ps);
	}
	KillBySelf(ps);
	KillByWall(ps);
}
void GameStart(pSnake ps)
{
	system("mode con cols=100 lines=30");
	system("title 贪吃蛇");
	HANDLE hOutput = GetStdHandle(STD_OUTPUT_HANDLE);
	CONSOLE_CURSOR_INFO CursorInfo;
	GetConsoleCursorInfo(hOutput, &CursorInfo);
	CursorInfo.bVisible = false;
	SetConsoleCursorInfo(hOutput, &CursorInfo);
	WelcomeToGame();
	CreatMap();
	InitSnake(ps);
	CreatFood(ps);
}
void GameRun(pSnake ps)
{
	PrintHelpInfo();
	do
	{
		SetPos(64, 10);
		printf("得分:%d ", ps->_score);
		printf("每个食物得分:%d", ps->_foodweight);
		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_RIGHT) && ps->_dir != LEFT)
		{
			ps->_dir = RIGHT;
		}
		else if (KEY_PRESS(VK_LEFT) && ps->_dir != RIGHT)
		{
			ps->_dir = LEFT;
		}
		else if (KEY_PRESS(VK_SPACE))
		{
			pause();
		}
		else if (KEY_PRESS(VK_ESCAPE))
		{
			ps->_status = END_NORMAL;
			break;
		}
		else if (KEY_PRESS(VK_F3))
		{
			if (ps->_sleeptime >= 50)
			{
				ps->_sleeptime -= 30;
				ps->_foodweight += 20;
			}
		}
		else if (KEY_PRESS(VK_F4))
		{
			if (ps->_sleeptime < 350)
			{
				ps->_sleeptime += 30;
				ps->_foodweight -= 20;
				if (ps->_sleeptime == 350)
				{
					ps->_foodweight = 1;
				}
			}
			
		}
		Sleep(ps->_sleeptime);
		SnakeMove(ps);

	} while (ps->_status == OK);
}
void GameEnd(pSnake ps)
{
	pSnakeNode cur = ps->_pbody;
	SetPos(24, 12);
	switch (ps->_status)
	{
	case KILL_BY_SELF:
		printf("你创了自己!,游戏结束\n");
		break;
	case KILL_BY_WALL:
		printf("你创了墙!game over\n");
		break;
	case END_NORMAL:
		printf("游戏正常结束!\n");
		break;
	}
	while (cur)
	{
		pSnakeNode del = cur;
		cur = cur->next;
		free(del);
	}
}

3.测试代码

#include<stdio.h>
#include"tcs.h"
void test()
{
	int ch = 0;
	srand((unsigned int)time(NULL));
	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, 27);
}
int main()
{
	setlocale(LC_ALL, "");
	test();
	return 0;
}

4.代码详解

1.头文件的解析

首先这一块是取库函数,和定义宏。

KEY_PRESS(VK)封装的是一个关于键盘是否被按下的函数。

这个函数的使用方法就是判断一个键盘按键是否被按下,如果按下它的返回值就是1不是就是0.所以用& 来判断是1还是0就能判断是否按下按键。 

 

这一块就是设置整体游戏函数的结构体,和定义一些参数。当然不习惯用enum定义,也可以用#define 来定义下面的方向以及游戏状态。 用typedef将 蛇身和整体的类型名缩写。

 

这一块就是等会需要实现的函数。 

2.源代码的解析 

 1.光标的位置封装函数

 

 

首先先说下COORD,他是一个结构体在WINDOWS API中定义的,里面存的变量是用来更改光标位置,为什么要改变光标的位置呢?改变光标位置你就可以在任何位置打印数据 。

第二个HANDLE就是一个变量,这个变量可以当做一个控制器,你可以给这个控制器变成任何一种工具,这里就获取了(GetStdHandle),标准输出(STD_OUTPUT_HANDLE),就是上面这个命令提示符。然后设置了此处的光标位置SetConsoleCursorPosition(houtput, pos)

GetStdHandle是获取一个控制器,下面的setconsolecursorposition根据字面意思就是设置光标位置。

 2.打印欢迎界面

void WelcomeToGame()
{
	SetPos(40, 15);
	printf("欢迎来到贪吃蛇\n");
	SetPos(40, 25);
	system("pause");
	system("cls");
	SetPos(25, 12);
	wprintf(L"⽤ ↑ . ↓ . ← . → 分别控制蛇的移动, F3为加速,F4为减速\n");
	SetPos(25, 13);
	printf("加速获得的分数更高\n");
	SetPos(40, 25);
	system("pause");
	system("cls");
}

 这一步就是用来打印欢迎界面以及介绍界面的

system("pause");这一步可以用来暂停,按一个键可以继续运行。
    system("cls");这一步是用来清屏的

3.打印整体的一个地图 

void CreatMap()
{
	SetPos(0, 0);
	for (int i = 0; i < 58; i = i + 2)
	{
		wprintf(L"%c", WALL);
	}
	SetPos(0, 26);
	for (int i = 0; i < 58; i= i + 2)
	{
		wprintf(L"%c", WALL);
	}
	for (int i = 1; i < 26; i++)
	{
		SetPos(0, i);
		wprintf(L"%c", WALL);
	}
	for (int i = 1; i < 26; i++)
	{
		SetPos(56, i);
		wprintf(L"%c", WALL);
	}
}

 首先游戏大小代码里设置的是58*26的就是58列26行。

然后setpos就是第一个函数,用来设定光标的位置,然后

关于wprintf因为在设计墙体蛇身这些图形时一个字符的空间是不够的,所以要用到wprintf,用来打印图形等双字符。且使用wprintf后面要在双引号前加上L表示打印宽字符。头文件是

#include<locale.h> 

4.蛇的初始化 (重要)

void InitSnake(pSnake ps)
{
	pSnakeNode cur = NULL;
	for (int i = 0; i < 5; i++)
	{
		cur = (pSnakeNode)malloc(sizeof(SnakeNode));
		if (cur == NULL)
		{
			perror("snakenode don't get:");
			return;
		}
		cur->next = NULL;
		cur->x = POS_X + i * 2;
		cur->y = POS_Y;
		if (ps->_pbody == NULL)
		{
			ps->_pbody = cur;
		}
		else
		{
			cur->next = ps->_pbody;
			ps->_pbody = cur;
		}
	}

	//print snake
	cur = ps->_pbody;
	while (cur)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%c", BODY);
		cur = cur->next;
	}
	//init snake data
	ps->_dir = RIGHT;
	ps->_foodweight = 100;
	ps->_score = 0;
	ps->_sleeptime = 200;
	ps->_status = OK;
}

关于初始化蛇,就是把snake 结构体的各项都初始化,还有把蛇身整体先打印出来。

关于蛇身的打印,可以根据你的想法打印出具体个数,for循环里更改就好。

首先关于这个蛇的实现我们用的是链表的知识,不知道的同学可以先去补一补链表。

对每个节点都设定好后,就是到while循环了,循环设定好的节点,把它打印在屏幕上。

这里用到setpos来找到需要打印的位置。

后面把snake结构体中的变量都初始化(根据自己喜欢调节就好)就可以进行下一步了。(这一步的蛇只是打印在屏幕上不能动。)

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->_pbody;
	while (cur)
	{
		if (cur->x == x && cur->y == y)
		{
			goto again;
		}
		cur = cur->next;
	}
	pSnakeNode pFood = (pSnakeNode)malloc(sizeof(SnakeNode));
	if (pFood == NULL)
	{
		perror("creat pfood failed,fool!\n");
		return;
	}
	else
	{
		pFood->x = x;
		pFood->y = y;
		SetPos(pFood->x, pFood->y);
		wprintf(L"%c", FOOD);
		ps->_pfood = pFood;
	}
}

因为玩的是贪吃蛇,我们现在初始化好蛇了,那接下来就是蛇的食物了。

用到rand()随机数。随机生成食物。

goto语句就是找到后面的名称的位置,跳到那个位置执行,(可以做到如果结果错误,重新打印食物)。其他的一看就懂了。

 5.打印边栏信息

void PrintHelpInfo()
{
	SetPos(64, 15);
	printf("不能创墙,也不能创自己");
	SetPos(64, 16);
	printf("用↑.↓.←.→分别控制蛇的移动.");
	SetPos(64, 17);
	printf("F3 为加速,F4 为减速\n");
	SetPos(64, 18);
	printf("ESC :退出游戏.space:暂停游戏.");
	SetPos(64, 20);
	printf("江子龙@版权");
}

6.设置用来暂停的函数 

void pause()
{
	while (1)
	{
		Sleep(300);
		if (KEY_PRESS(VK_SPACE))
		{
			break;
		}
	}
}

 这个函数主要用来,当你按下space的时候,给计算机一个sleep的死循环。

7.  3个函数,关于走的下一步是不是食物 

int NextIsFood(pSnakeNode psn, pSnake ps)
{
	return(psn->x == ps->_pfood->x) && (psn->y == ps->_pfood->y);
}
void EatFood(pSnakeNode psn, pSnake ps)
{
	psn->next = ps->_pbody;
	ps->_pbody = psn;
	pSnakeNode cur = ps->_pbody;
	while (cur)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%c", BODY);
		cur = cur->next;
	}
	ps->_score += ps->_foodweight;
	free(ps->_pfood);
	CreatFood(ps);
}
void NotFood(pSnakeNode psn, pSnake ps)
{
	psn->next = ps->_pbody;
	ps->_pbody = psn;
	pSnakeNode cur = ps->_pbody;
	while (cur->next->next)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%c", BODY);
		cur = cur->next;
	}
	SetPos(cur->next->x, cur->next->y);
	printf("  ");
	free(cur->next);
	cur->next = NULL;
}

第一个返回的是int 是为了等会在主体函数中判断真假,如果为真为1则执行EatFood,为假为0则执行NotFood。 (贪吃蛇变长的条件)

第二个EatFood 如果下一个点是食物的话,贪吃蛇就变长一格,用的是头插。

第三个NotFood 如果下一点不是食物的话,贪吃蛇就整个向前走一格,就是头插下一个空白格,然后free掉尾巴最后一格。

 8.关于蛇是怎么死的

int KillByWall(pSnake ps)
{
	if ((ps->_pbody->x == 0) || (ps->_pbody->x == 56) || (ps->_pbody->y == 0) || (ps->_pbody->y == 26))
	{
		ps->_status = KILL_BY_WALL;
		return 1;
	}
	return 0;
}
int KillBySelf(pSnake ps)
{
	pSnakeNode cur = ps->_pbody->next;
	while (cur)
	{
		if ((ps->_pbody->x == cur->x) && (ps->_pbody->y == cur->y))
		{
			ps->_status = KILL_BY_SELF;
			return 1;
		}
		cur = cur->next;
	}
	return 0;
}

接下来的这两个代码就是用来判断,你的蛇是怎么死的。

第一个代码判断出来的就是如果 蛇的头与墙的位置出现在了一起,那就把状态改变蛇就死了

第二个代码判断的是如果蛇和自己的身体的某一个节点的位置相同,那蛇的状态就是死了。 

 9.蛇的移动(重点)

void SnakeMove(pSnake ps)
{
	pSnakeNode pNextNode = (pSnakeNode)malloc(sizeof(SnakeNode));
	if (pNextNode == NULL)
	{
		perror("snake move open falied");
		return;
	}

	switch (ps->_dir)
	{
	case UP:
	{
		pNextNode->y = ps->_pbody->y - 1;
		pNextNode->x = ps->_pbody->x;
	}
		break;
	case DOWN:
	{
		pNextNode->y = ps->_pbody->y + 1;
		pNextNode->x = ps->_pbody->x;
	}
		break;
	case LEFT:
	{
		pNextNode->y = ps->_pbody->y ;
		pNextNode->x = ps->_pbody->x - 2;
	}
		break;
	case RIGHT:
	{
		pNextNode->y = ps->_pbody->y;
		pNextNode->x = ps->_pbody->x + 2;
	}
		break;
	default:
		break;
	}
	if (NextIsFood(pNextNode, ps))
	{
		EatFood(pNextNode, ps);
	}
	else
	{
		NotFood(pNextNode, ps);
	}
	KillBySelf(ps);
	KillByWall(ps);
}

蛇的移动代码量偏长,首先定义好蛇的移动方向上下左右,是怎么改变的。

因为蛇的节点的宽字符,所以在横坐标改变时是两倍改变,而纵坐标为单倍改变即可。

然后在进行判断蛇移动的下一个位置是否是食物,是食物则执行前面的三个函数。

最后判断蛇是否死亡。

 10.游戏的运行(游戏的逻辑)

void GameStart(pSnake ps)
{
	system("mode con cols=100 lines=30");
	system("title 贪吃蛇");
	HANDLE hOutput = GetStdHandle(STD_OUTPUT_HANDLE);
	CONSOLE_CURSOR_INFO CursorInfo;
	GetConsoleCursorInfo(hOutput, &CursorInfo);
	CursorInfo.bVisible = false;
	SetConsoleCursorInfo(hOutput, &CursorInfo);
	WelcomeToGame();
	CreatMap();
	InitSnake(ps);
	CreatFood(ps);
}
void GameRun(pSnake ps)
{
	PrintHelpInfo();
	do
	{
		SetPos(64, 10);
		printf("得分:%d ", ps->_score);
		printf("每个食物得分:%d", ps->_foodweight);
		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_RIGHT) && ps->_dir != LEFT)
		{
			ps->_dir = RIGHT;
		}
		else if (KEY_PRESS(VK_LEFT) && ps->_dir != RIGHT)
		{
			ps->_dir = LEFT;
		}
		else if (KEY_PRESS(VK_SPACE))
		{
			pause();
		}
		else if (KEY_PRESS(VK_ESCAPE))
		{
			ps->_status = END_NORMAL;
			break;
		}
		else if (KEY_PRESS(VK_F3))
		{
			if (ps->_sleeptime >= 50)
			{
				ps->_sleeptime -= 30;
				ps->_foodweight += 20;
			}
		}
		else if (KEY_PRESS(VK_F4))
		{
			if (ps->_sleeptime < 350)
			{
				ps->_sleeptime += 30;
				ps->_foodweight -= 20;
				if (ps->_sleeptime == 350)
				{
					ps->_foodweight = 1;
				}
			}
			
		}
		Sleep(ps->_sleeptime);
		SnakeMove(ps);

	} while (ps->_status == OK);
}
void GameEnd(pSnake ps)
{
	pSnakeNode cur = ps->_pbody;
	SetPos(24, 12);
	switch (ps->_status)
	{
	case KILL_BY_SELF:
		printf("你创了自己!,游戏结束\n");
		break;
	case KILL_BY_WALL:
		printf("你创了墙!game over\n");
		break;
	case END_NORMAL:
		printf("游戏正常结束!\n");
		break;
	}
	while (cur)
	{
		pSnakeNode del = cur;
		cur = cur->next;
		free(del);
	}
}

 每个游戏的运行都分为三步,开始,玩,结束。

这里也一样,游戏的开始就是用上面的函数进行游戏的初始化,菜单的打印,以及初始化蛇之类的开始函数。

第二步玩,这一步要用到之前封装的宏 KEY_PRESS( VK ),用来判断这个按键是否被按下,被按下就执行这个逻辑。同时贪吃蛇的一部分按键是不能重合的,像上和下是不能同时进行的,所以要进行判断。最后加上snakemove等函数,蛇就能动起来。

第三步结束,这一步就是判断你的游戏时因为什么结束的。然后再把蛇的节点全部free,以备开启下一把

11.关于整体游戏的测试代码

void test()
{
	int ch = 0;
	srand((unsigned int)time(NULL));
	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, 27);
}
int main()
{
	setlocale(LC_ALL, "");
	test();
	return 0;
}

setlocale 是用来适应当地环境的。(用来打印宽字符)

srand函数,是为了让rand 函数的值不是伪随机值,而是随着时间更改。

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

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

相关文章

BUUCTF——刮开有奖

打开程序&#xff1a; 就一个这个玩意儿&#xff0c;没有输入框&#xff0c;没有啥的&#xff0c;打开IDA反编译一下&#xff1a; 直接找到WinMain&#xff0c;发现里面只有一个对话框API&#xff08;如果只有一个对话框&#xff0c;那真就没有输入框了&#xff09;&#xff0…

Pytorch从零开始实战09

Pytorch从零开始实战——YOLOv5-Backbone模块实现 本系列来源于365天深度学习训练营 原作者K同学 文章目录 Pytorch从零开始实战——YOLOv5-Backbone模块实现环境准备数据集模型选择开始训练可视化模型预测总结 环境准备 本文基于Jupyter notebook&#xff0c;使用Python3.…

养老院信息展示预约小程序的效果如何

老龄化速度加快及快节奏时代&#xff0c;银发群体的老年生活&#xff0c;儿女往往难以照顾&#xff0c;养老院成为不少家庭或个人的选择&#xff0c;靠谱机构往往能带给老人丰富多彩的生活。 而在高需求的同时&#xff0c;无论对需求者还是养老院本身都存在一定难题&#xff1…

Leetcode-101 对称二叉树

递归&#xff1a;主要思想&#xff1a;对称二叉树是左子树的左孩子右子树的右孩子&#xff0c;左子树的右孩子右子树的左孩子&#xff0c;递归实现思路较为清晰 /*** Definition for a binary tree node.* public class TreeNode {* int val;* TreeNode left;* Tr…

为什么要做MBTI职业性格测试?

MBTI职业性格测试是一种成熟的人格测评工具&#xff0c;基于荣格理论发展而来&#xff0c;将人的性格分为16种类型&#xff0c;或内向。或外向&#xff0c;或注重情感&#xff0c;或注重感知。 每种性格各有长处和不足&#xff0c;通过应用mbti职业性格测试&#xff0c;可以方…

字节和美团软件测试面试1000问(含文档)

一、Linux系统应用和环境配置 1、Linux系统的操作命令给我说10个&#xff0c;一般用什么工具远程连接Linux服务器&#xff1f; 2、Linux中的日志存储在哪里&#xff1f;怎么查看日志内容&#xff1f; 3、Linux中top和ps命令的区别&#xff1f; 4、Linux命令运行的结果如何写…

synchronized的原理和Callable接口

目录 ♫synchronized原理 ♪锁升级 ♪锁优化 ♫Callable接口 ♫synchronized原理 我们知道synchronized锁可以控制多个线程对共享资源的访问&#xff0c;两个线程针对同一变量访问就会产生阻塞等待。而synchronized锁并不是一成不变的&#xff0c;它会根据情况进行一次升级。…

自动化测试系列 —— UI自动化测试

UI 测试是一种测试类型&#xff0c;也称为用户界面测试&#xff0c;通过该测试&#xff0c;我们检查应用程序的界面是否工作正常或是否存在任何妨碍用户行为且不符合书面规格的 BUG。了解用户将如何在用户和网站之间进行交互以执行 UI 测试至关重要&#xff0c;通过执行 UI 测试…

自适应AI chatGPT智能聊天创作官网html源码/最新AI创作系统/ChatGPT商业版网站源码

源码简介&#xff1a; 自适应AI chatGPT智能聊天创作官网html源码&#xff0c;这是最新AI创作系统&#xff0c;作为ChatGPT商业版网站源码&#xff0c;它是支持创作、编写、翻译、写代码等。是一个智能聊天系统项目源码。 注意&#xff1a;这个只是网站html源码&#xff0c;要…

8年经验之谈 —— 如何用 JMeter 编写性能测试脚本?

Apache JMeter 应该是应用最广泛的性能测试工具。怎么用 JMeter 编写性能测试脚本&#xff1f; 1. 编写 HTTP 性能测试脚本 STEP 1. 添加 HTTP 请求 img STEP 2. 了解配置信息 HTTP 请求各项信息说明&#xff08;以 JMeter 5.1 为例&#xff09;。 如下图所示&#xff1a;…

第三章《补基础:不怕学不懂概率统计》笔记

3.1 什么是概率 概率亦称“或然率”&#xff0c;它反映随机事件出现的可能性大小&#xff0c;在现实生活中有着极其普遍的应用。 3.1.1 最简单的概率的例子 3.1.2 概率论与数理统计的关系 概率论与数理统计的关系可以概括为&#xff0c;概率论是数理统计的理论基础&#xf…

VSCode + PlatformIO ESP32开发环境配置(离线版5分钟搞定)

文章目录 安装python1. 打开应用商店2. 应用商店搜索python3. 安装python4. python安装完成5. 打开命令提示符6. 验证安装结果7. 更新pip源为国内源 安装VSCode下载VSCode安装Vscode安装简体中文插件安装VSCode platformio插件安装Prettier - Code formatter插件 &#xff08;建…

图书销售数据大屏可视化【可视化项目案例-03】

🎉🎊🎉 你的技术旅程将在这里启航! 🚀🚀 本文选自专栏:可视化技术专栏100例 可视化技术专栏100例,包括但不限于大屏可视化、图表可视化等等。订阅专栏用户在文章底部可下载对应案例源码以供大家深入的学习研究。 🎓 每一个案例都会提供完整代码和详细的讲解,不…

【C++那些事儿】类与对象(2)

君兮_的个人主页 即使走的再远&#xff0c;也勿忘启程时的初心 C/C 游戏开发 Hello,米娜桑们&#xff0c;这里是君兮_&#xff0c;我之前看过一套书叫做《明朝那些事儿》&#xff0c;把本来枯燥的历史讲的生动有趣。而C作为一门接近底层的语言&#xff0c;无疑是抽象且难度颇…

合同审查---财务条款、合同形式与生效审查

1.合同主体 1人 廖 2.财务条款、合同形式与生效 1人 黄 3.履行、验收、知识产权、不可抗力 1人 詹 4.违约责任、争议解决、保密、法律引用 1人 王 代码规范&#xff1a; 1.代码函数的层级 各审查点在json中分为3级层级&#xff0c;但用python写规则的时候&#xff0c;1级层级为…

2023年成为优秀自动化测试工程师的 7 个步骤!

“测试自动化测试工程师可以将你从充满代码的世界中拯救出来。”企业完全同意这一说法&#xff0c;这就是您在自动化测试行业中看到大量就业机会的原因。我在 Quora 上收到了很多与自动化测试中的职业选择相关的答案请求&#xff0c;以及人们如何在有或没有手动测试经验的情况下…

JVM源码剖析之软、弱、虚引用的处理细节

目录 写在前面&#xff1a; 源码剖析&#xff1a; Java层面&#xff1a; JVM层面&#xff1a; 使用危险点&#xff1a; 总结&#xff1a; 版本信息&#xff1a; jdk版本&#xff1a;jdk8u40 垃圾回收器&#xff1a;Serial new/old 写在前面&#xff1a; 不同的垃圾回收…

第23章(下)_索引原理剖析

文章目录 索引实现索引存储B树为什么 MySQL InnoDB 选择 B 树作为索引的数据结构&#xff1f;B 树层高问题关于自增id最左匹配原则覆盖索引索引下推innodb体系结构Buffer poolchange buffer 总结 索引实现 索引存储 innodb 由段、区、页组成。段分为数据段、索引段、回滚段等…

摊位展示预约小程序的作用有哪些

无论市场还是街边&#xff0c;小摊小贩往往很多&#xff0c;组成了丰富多彩的线下购物环境&#xff0c;而在实际发展中&#xff0c;摊位的需求度很高&#xff0c;但由于种种原因&#xff0c;导致在实际发展中&#xff0c;也有一定难题&#xff1a; 1、摊位预约难、信息查看难 …

js 求数组中的对象某个属性和

可以直接看下效果 代码&#xff1a; <script>let list [{num: 1,price: 10,},{num: 2,price: 10,},{num: 3,price: 10,},{num: 4,price: 10,},]// for循环 求总数和 num的和let num 0for (let i 0; i < list.length; i) {num list[i].num}console.log(第一种&am…