贪吃蛇项目(基于C语言和数据结构中的链表)

news2025/1/19 7:58:27

建立文件

首先先建立3个文件。

Snake.h 函数的声明

Snake.c 函数的定义

Test.c     贪吃蛇的测试   

 分析项目

我们分析这整个项目

 建立节点

首先在我们实现游戏开始的部分之前,我们要先创建贪吃蛇的节点,再由此创建整个贪吃蛇所包含的一些信息:

#pragma once


//蛇身节点
typedef struct SnakeNode
{
	int x;
	int y;
	struct SnakeNode* next;
}SnakeNode,* pSnakeNode;
//SnakeNode 是struct SnakeNode的重命名
//pSnakeNode 是struct SnakeNode *的重命名


//我们再根据此来创建整条蛇
typedef struct Snake
{
	pSnakeNode pSnake;//维护整条蛇的指针
	pSnakeNode pFood; //维护食物的指针
	int score;//当前的分数
	int foodWeight;//食物的权重
	int SleepTime;//休眠时间(其实就是你肉眼能看到的一眨一眨的)
	enum DIRECTION Dir;//蛇重生默认方向
	enum GAMESTATUS Status;//游戏状态
}Snake,*pSnake;

建立贪吃蛇所包含的一些信息 

我们枚举一下这个贪吃蛇中所有的一些状态:

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

//游戏状态
enum GAME_STATUS
{
	OK=1,//正常运行
	ESC, //按了ESC键退出,正常退出
	KILL_BY_WALL,//撞墙
	KILL_BY_SELF //撞到自身
};

写Test.c函数 

然后我们就能书写测试函数中的内容了

根据这张图来写

 

#define _CRT_SECURE_NO_WARNINGS 1
#include "Snake.h"
#include <locale.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;

SetPos 

1.其中的SetPos:封装⼀个设置光标位置的函数,他的头文件为#include <window.h>

//设置光标的坐标
void SetPos(short x, short y)
{
	COORD pos = { x,y };
	HANDLE hOutput = NULL;
	//获取标准输出的句柄(⽤来标识不同设备的数值)
	hOutput = GetStdHandle(STD_OUTPUT_HANDLE);
	//设置标准输出上光标的位置为pos
	SetConsoleCursorPosition(hOutput, pos);
}

setlocale 

2.其中的setlocale为:

<locale.h>提供的函数⽤于控制C标准库中对于不同的地区会产⽣不⼀样⾏为的部分。
setlocale 函数⽤于修改当前地区,可以针对⼀个类项修改,也可以针对所有类项。
setlocale 的第⼀个参数可以是前⾯说明的类项中的⼀个,那么每次只会影响⼀个类项,如果第⼀个参 数是LC_ALL,就会影响所有的类项。
C标准给第⼆个参数仅定义了2种可能取值:"C"和" "。

当地区设置为"C"时,库函数按正常⽅式执⾏,⼩数点是⼀个点。
当程序运⾏起来后想改变地区,就只能显⽰调⽤setlocale函数。⽤" "作为第2个参数,调⽤setlocale 函数就可以切换到本地模式,这种模式下程序会适应本地环境。⽐如:切换到我们的本地模式后就⽀ 持宽字符(汉字)的输出等。

游戏开始:GameStart 函数

接下来我们实现GameStart函数 

void GameStart(pSnake ps)
{
	//设置控制台窗口的大小,50行,150列
	system("mode con cols=150 lines=50");
	//设置窗口名称
	system("title 贪吃蛇大作战");
	//获取标准输出的句柄(⽤来标识不同设备的数值)
	HANDLE hOutput = GetStdHandle(STD_OUTPUT_HANDLE);
	//隐藏光标操作
	CONSOLE_CURSOR_INFO CursorInfo;
	GetConsoleCursorInfo(hOutput, &CursorInfo);//获取控制台光标信息
	CursorInfo.bVisible = false; //隐藏控制台光标
	SetConsoleCursorInfo(hOutput, &CursorInfo);//设置控制台光标状态

	//打印欢迎界面
	WelcomTOGame();
	//打印地图
	CreateMap();
	//初始化蛇
	InitSnkae(ps);
	//创造第一个食物
	CreateFood(ps);
}

我们再实现GameStart的

//打印欢迎界面
   WelcomTOGame();
    //打印地图
    CreateMap();
    //初始化蛇
    InitSnkae(ps);
    //创造第一个食物
    CreateFood(ps);

欢迎界面 WelcomTOGame();

void WelcomTOGame()
{
	SetPos(80, 30);
	printf("欢迎来到贪吃蛇大作战");
	SetPos(80, 50);
	system("pause");
	system("cls");
	SetPos(50, 25);
	printf("⽤ ↑.↓.←.→ 分别控制蛇的移动, F1为加速,F2为减速\n");
	SetPos(50, 26);
	printf("加速是给勇士的奖励.");
	SetPos(40, 25);
	system("pause");
	system("cls");
	getchar();//此时加一个getchar()是为了让他在这里停下,否则我们看不到这个过程,只是为了更
              //好的测试
}

我们这时候运行一下程序看下结果 

 这里的 ↑.↓.←.→是用的搜狗输入法符号大全里的

 

 打印地图:CreateMap(); 

void CreateMap()
{
	int i = 0;
	//上下左右
	SetPos(0, 0);
	for (i = 0; i < 114; i += 2)
	{
		wprintf(L"%c", WALL);
	}
	SetPos(0, 52);
	for (i = 0; i < 114; i += 2)
	{
		wprintf(L"%c", WALL);
	}
	for(i = 1; i < 52; i++)
	{
		SetPos(0, i);
		wprintf(L"%c", WALL);
	}
	for (i = 1; i < 52; i++)
	{
		SetPos(112, i);
		wprintf(L"%c", WALL);
	}
	//打印提⽰信息
	SetPos(128, 30);
	printf("不能穿墙,不能咬自己\n");
	SetPos(128, 32);
	printf("利用↑.↓.←.→分别控制蛇的移动.");
	SetPos(128, 34);
	printf("F1 为加速,F2 为减速\n");
	SetPos(128, 36);
	printf("ESC :退出游戏.space:暂停游戏.");
	SetPos(128, 40);
	printf("@你小子别偷懒了");
	
}

 这里的WALL在.h文件中用宏定义说明一下

#define WALL L'◇'

这个时候我们再次运行一次代码

跟上次不同的是:这次多了一个规则的地图 

 初始化蛇 : InitSnkae(ps);
  

void InitSnake(pSnake ps)
{
	//创建一个单独的节点,这是为了创建蛇的身体
	pSnakeNode cur = NULL;
	int i = 0;
	//创建蛇身节点,并初始化坐标,利用头插法
	for (i = 0; i < 5; i++)
	{
		cur = (pSnakeNode)malloc(sizeof(SnakeNode));
		if (cur == NULL)
		{
			perror("malloc fail");
			return;
		}
		//如果cur不为空,就设置坐标
		cur->next = NULL;
		cur->x = POS_X + i * 2;
		cur->y = POS_Y;

		//然后给本游戏的蛇进行设置
		if (ps->pSnake == NULL)
		{
			ps->pSnake = cur;
		}
		else
		{
			cur->next = ps->pSnake;
			ps->pSnake = cur;
		}
	}
	//打印蛇
	cur = ps->pSnake;
	while (cur)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%c", BODY);
		cur = cur->next;
	}
	ps->SleepTime = 200;
	ps->score = 0;
	ps->status = OK;
	ps->dir = RIGHT;
	ps->foodWeight = 10;
	getchar();
}

 我们再次运行:此时就有蛇的存在了

 

 创造第一个食物:CreateFood(ps);  

到这里我们还差创建食物

void CreateFood(pSnake ps)
{
	int x = 0;
	int y = 0;

again:
	//x坐标为2的倍数,如果他一直不为2的倍数就一直算
	do
	{
		x = rand() % 109 + 2;
		y = rand() % 52 + 1;
	} while (x % 2 != 0);

	pSnakeNode cur = ps->pSnake;//获取指向蛇头的指针
	//食物不能和蛇身冲突
	while (cur)
	{
		if (cur->x == x && cur->y == y)
		{
			goto again;
		}
		cur = cur->next;
	}

	pSnakeNode pFood = (pSnakeNode)malloc(sizeof(SnakeNode));//创建食物
	if (pFood == NULL)
	{
		perror("malloc fail");
		return;
	}

		pFood->x = x;
        pFood->y = y;

        ps->pFood = pFood;
        SetPos(x, y);
        wprintf(L"%lc", FOOD);
}

 这个时候我们再次运行代码

食物出现了,也就是说我们游戏开始的过程完成了,接下来就是游戏中的代码了。

游戏中:GameRun 函数 

 

 

void GameRun(pSnake ps)
{
	

	do
	{
		//当前的分数情况
		SetPos(124, 20);
		printf("总分:%5d\n", ps->score);
		SetPos(124, 22);
		printf("食物的分值:%02d\n", ps->FoodWeight);
		//检测按键
		//上、下、左、右、ESC、空格、F1、F2
		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_ESCAPE))
		{
			ps->status = ESC;
			break;
		}
		else if (KEY_PRESS(VK_SPACE))
		{
			//游戏要暂定
			pause();//暂定和回复暂定
		}
		else if (KEY_PRESS(VK_F1))
		{
			if (ps->SleepTime >= 80)
			{
				ps->SleepTime -= 30;
				ps->FoodWeight += 2;
			}
		}
		else if (KEY_PRESS(VK_F2))
		{
			if (ps->FoodWeight > 2)
			{
				ps->SleepTime += 30;
				ps->FoodWeight -= 2;
			}
		}
		//走一步
		SnakeMove(ps);

		//睡眠一下
		Sleep(ps->SleepTime);

	} while (ps->status == OK);
}

上述代码有pause函数和虚拟按键,和SnakeMove函数没有介绍到

pause函数

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

虚拟按键 

获取按键情况,GetAsyncKeyState的函数:
将键盘上每个键的虚拟键值传递给函数,函数通过返回值来分辨按键的状态。
GetAsyncKeyState 的返回值是short类型,在上⼀次调⽤ GetAsyncKeyState 函数后,如果
返回的16位的short数据中,最⾼位是1,说明按键的状态是按下,如果最⾼是0,说明按键的状态是抬
起;如果最低位被置为1则说明,该按键被按过,否则为0。
如果我们要判断⼀个键是否被按过,可以检测GetAsyncKeyState返回值的最低值是否为1.

在程序中需要添加宏定义:#define KEY_PRESS(VK) ( (GetAsyncKeyState(VK) & 0x1) ? 1 : 0 )

SnakeMove函数

void SnakeMove(pSnake ps)
{
	pSnakeNode pNext = (pSnakeNode)malloc(sizeof(SnakeNode));
	if (pNext == NULL)
	{
		perror("SnakeMove()::malloc()");
		return;
	}
	pNext->next = NULL;

	switch (ps->dir)
	{
	case UP:
		pNext->x = ps->pSnake->x;
		pNext->y = ps->pSnake->y - 1;
		break;
	case DOWN:
		pNext->x = ps->pSnake->x;
		pNext->y = ps->pSnake->y + 1;
		break;
	case LEFT:
		pNext->x = ps->pSnake->x - 2;
		pNext->y = ps->pSnake->y;
		break;
	case RIGHT:
		pNext->x = ps->pSnake->x + 2;
		pNext->y = ps->pSnake->y;
		break;
	}

	//下一个坐标处是否是食物
	if (NextIsFood(ps, pNext))
	{
		//是食物就吃掉
		EatFood(ps, pNext);
	}
	else
	{
		//不是食物就正常一步
		NotEatFood(ps, pNext);
	}

	//检测撞墙
	KillByWall(ps);

	//检测撞到自己
	KillBySelf(ps);
}

 这里面又有5个函数不知道:NextIsFood        EatFood        NotEatFood    KillByWall     KillBySelf      

 NextIsFood

//pSnakNode psn 是下一个节点的地址
//pSnake ps 维护蛇的指针
int NextIsFood(pSnake ps, pSnakeNode pNext)
{
	if (ps->pFood->x == pNext->x && ps->pFood->y == pNext->y)
		return 1;//下一个坐标处是食物
	else
		return 0;
}

 EatFood 

void EatFood(pSnake ps, pSnakeNode pNext)
{
	pNext->next = ps->pSnake;
	ps->pSnake = pNext;

	//打印蛇
	pSnakeNode cur = ps->pSnake;
	while (cur)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%lc", BODY);
		cur = cur->next;
	}

	ps->score += ps->FoodWeight;

	//释放旧的食物
	free(ps->pFood);
	//新建食物
	CreateFood(ps);
}

 NotEatFood 

void NotEatFood(pSnake ps, pSnakeNode pNext)
{
	//头插法
	pNext->next = ps->pSnake;
	ps->pSnake = pNext;

	//释放尾结点
	pSnakeNode cur = ps->pSnake;
	while (cur->next->next)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%lc", BODY);
		cur = cur->next;
	}
	//将尾节点的位置打印成空白字符
	SetPos(cur->next->x, cur->next->y);
	printf("  ");

	free(cur->next);
	cur->next = NULL;//易错
}

 KillByWall 

//检测是否撞墙
void KillByWall(pSnake ps)
{
	if (ps->pSnake->x == 0 ||
		ps->pSnake->x == 112 ||
		ps->pSnake->y == 0 ||
		ps->pSnake->y == 52)
	{
		ps->status = KILL_BY_WALL;
	}
}

  KillBySelf   

//检测是否撞自己
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;
			return;
		}
		cur = cur->next;
	}
}

游戏结束:GameEnd函数 

此时程序总体还差一个GameEnd函数

void GameEnd(pSnake ps)
{
	SetPos(30, 24);
	switch (ps->status)
	{
	case ESC:
		printf("主动退出游戏,正常退出\n");
		break;
	case KILL_BY_WALL:
		printf("很遗憾,撞墙了,游戏结束\n");
		break;
	case KILL_BY_SELF:
		printf("很遗憾,咬到自己了,游戏结束\n");
		break;
	}

	pSnakeNode cur = ps->pSnake;

	//释放蛇身的节点
	while (cur)
	{
		pSnakeNode del = cur;
		cur = cur->next;
		free(del);
	}
}

最后我们给上最终代码

最终代码

Snake.h

#pragma once
#include <locale.h>
#include <stdlib.h>
#include <stdio.h>
#include <windows.h>
#include <stdbool.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 )
#define KEY_PRESS(VK)  ( (GetAsyncKeyState(VK) & 0x1) ? 1 : 0 )


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

//游戏状态
enum GAME_STATUS
{
	OK=1,//正常运行
	ESC, //按了ESC键退出,正常退出
	KILL_BY_WALL,//撞墙
	KILL_BY_SELF //撞到自身
};



//蛇身节点
typedef struct SnakeNode
{
	int x;
	int y;
	struct SnakeNode* next;
}SnakeNode,* pSnakeNode;
//SnakeNode 是struct SnakeNode的重命名
//pSnakeNode 是struct SnakeNode *的重命名


//我们再根据此来创建整条蛇
typedef struct Snake
{
	pSnakeNode pSnake;//维护整条蛇的指针
	pSnakeNode pFood; //维护食物的指针
	int score;//当前的分数
	int FoodWeight;//食物的权重
	int SleepTime;//休眠时间(其实就是你肉眼能看到的一眨一眨的)
	enum DIRECTION dir;//蛇重生默认方向
	enum GAMESTATUS status;//游戏状态
}Snake,* pSnake;


//游戏开始前的初始化
void GameStart(pSnake ps);
//游戏运⾏过程
void GameRun(pSnake ps);
//游戏结束
void GameEnd(pSnake ps);
//设置光标的坐标
void SetPos(short x, short y);
//欢迎界⾯
void WelcomToGame();
//创建地图
void CreateMap();
//初始化蛇
void InitSnake(pSnake ps);
//创建⻝物
void CreateFood(pSnake ps);
//暂停响应
void pause();
//下⼀个节点是⻝物
int NextIsFood(pSnake ps, pSnakeNode pNext);

//下一步要走的位置处就是食物,就吃掉食物
void EatFood(pSnake ps, pSnakeNode pNext);
//下一步要走的位置处不是食物,不吃食物
void NotEatFood(pSnake ps, pSnakeNode pNext);
//撞墙检测
void KillByWall(pSnake ps);
//撞⾃⾝检测
void KillBySelf(pSnake ps);
//蛇的移动
void SnakeMove(pSnake ps);
//游戏初始化
void GameStart(pSnake ps);
//游戏运⾏
void GameRun(pSnake ps);
//游戏结束
void GameEnd(pSnake ps);

Snake.c

#define _CRT_SECURE_NO_WARNINGS 1
#include "Snake.h"

//设置光标的坐标
void SetPos(int x, int y)
{
	COORD pos = { x,y };
	HANDLE hOutput = NULL;
	//获取标准输出的句柄(⽤来标识不同设备的数值)
	hOutput = GetStdHandle(STD_OUTPUT_HANDLE);
	//设置标准输出上光标的位置为pos
	SetConsoleCursorPosition(hOutput, pos);
}
void CreateMap()
{
	int i = 0;
	//上下左右
	SetPos(0, 0);
	for (i = 0; i < 114; i += 2)
	{
		wprintf(L"%c", WALL);
	}
	SetPos(0, 52);
	for (i = 0; i < 114; i += 2)
	{
		wprintf(L"%c", WALL);
	}
	for(i = 1; i < 52; i++)
	{
		SetPos(0, i);
		wprintf(L"%c", WALL);
	}
	for (i = 1; i < 52; i++)
	{
		SetPos(112, i);
		wprintf(L"%c", WALL);
	}
	//打印提⽰信息
	SetPos(128, 30);
	printf("不能穿墙,不能咬自己\n");
	SetPos(128, 32);
	printf("利用↑.↓.←.→分别控制蛇的移动.");
	SetPos(128, 34);
	printf("F1 为加速,F2 为减速\n");
	SetPos(128, 36);
	printf("ESC :退出游戏.space:暂停游戏.");
	SetPos(128, 40);
	printf("@你小子别偷懒了");
	
}
void WelcomToGame()
{
	SetPos(80, 30);
	printf("欢迎来到贪吃蛇大作战");
	SetPos(80, 50);
	system("pause");
	system("cls");
	SetPos(50, 25);
	printf("⽤ ↑.↓.←.→ 分别控制蛇的移动, F1为加速,F2为减速\n");
	SetPos(50, 26);
	printf("加速是给勇士的奖励.");
	SetPos(40, 25);
	system("pause");
	system("cls");
}
void InitSnake(pSnake ps)
{
	//创建一个单独的节点,这是为了创建蛇的身体
	pSnakeNode cur = NULL;
	int i = 0;
	//创建蛇身节点,并初始化坐标,利用头插法
	for (i = 0; i < 5; i++)
	{
		cur = (pSnakeNode)malloc(sizeof(SnakeNode));
		if (cur == NULL)
		{
			perror("malloc fail");
			return;
		}
		//如果cur不为空,就设置坐标
		cur->next = NULL;
		cur->x = POS_X + i * 2;
		cur->y = POS_Y;

		//然后给本游戏的蛇进行设置
		if (ps->pSnake == NULL)
		{
			ps->pSnake = cur;
		}
		else
		{
			cur->next = ps->pSnake;
			ps->pSnake = cur;
		}
	}
	//打印蛇
	cur = ps->pSnake;
	while (cur)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%c", BODY);
		cur = cur->next;
	}
	ps->SleepTime = 200;
	ps->score = 0;
	ps->status = OK;
	ps->dir = RIGHT;
	ps->FoodWeight = 10;
	ps->pFood = NULL;
}
void CreateFood(pSnake ps)
{
	int x = 0;
	int y = 0;

again:
	//x坐标为2的倍数,如果他一直不为2的倍数就一直算
	do
	{
		x = rand() % 109 + 2;
		y = rand() % 52 + 1;
	} while (x % 2 != 0);

	pSnakeNode cur = ps->pSnake;//获取指向蛇头的指针
	//食物不能和蛇身冲突
	while (cur)
	{
		if (cur->x == x && cur->y == y)
		{
			goto again;
		}
		cur = cur->next;
	}

	pSnakeNode pFood = (pSnakeNode)malloc(sizeof(SnakeNode));//创建食物
	if (pFood == NULL)
	{
		perror("malloc fail");
		return;
	}

		pFood->x = x;
		pFood->y = y;

		ps->pFood = pFood;
		SetPos(x, y);
		wprintf(L"%lc", FOOD);

	
	
}
void GameStart(pSnake ps)
{
	//设置控制台窗口的大小,60行,200列
	system("mode con cols=200 lines=60");
	//设置窗口名称
	system("title 贪吃蛇大作战");
	//获取标准输出的句柄(⽤来标识不同设备的数值)
	HANDLE hOutput = GetStdHandle(STD_OUTPUT_HANDLE);
	//隐藏光标操作
	CONSOLE_CURSOR_INFO CursorInfo;
	GetConsoleCursorInfo(hOutput, &CursorInfo);//获取控制台光标信息
	CursorInfo.bVisible = false; //隐藏控制台光标
	SetConsoleCursorInfo(hOutput, &CursorInfo);//设置控制台光标状态

	//打印欢迎界面
	WelcomToGame();
	//打印地图
	CreateMap();
	//初始化蛇
	InitSnake(ps);
	//创造第一个食物
	CreateFood(ps);
}
void pause()
{
	while (1)
	{
		Sleep(100);
		if (KEY_PRESS(VK_SPACE))
		{
			break;
		}
	}
}

int NextIsFood(pSnake ps, pSnakeNode pNext)
{
	if (ps->pFood->x == pNext->x && ps->pFood->y == pNext->y)
		return 1;//下一个坐标处是食物
	else
		return 0;
}

void EatFood(pSnake ps, pSnakeNode pNext)
{
	pNext->next = ps->pSnake;
	ps->pSnake = pNext;

	//打印蛇
	pSnakeNode cur = ps->pSnake;
	while (cur)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%lc", BODY);
		cur = cur->next;
	}

	ps->score += ps->FoodWeight;

	//释放旧的食物
	free(ps->pFood);
	//新建食物
	CreateFood(ps);
}

void NotEatFood(pSnake ps, pSnakeNode pNext)
{
	//头插法
	pNext->next = ps->pSnake;
	ps->pSnake = pNext;

	//释放尾结点
	pSnakeNode cur = ps->pSnake;
	while (cur->next->next)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%lc", BODY);
		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 == 112 ||
		ps->pSnake->y == 0 ||
		ps->pSnake->y == 52)
	{
		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;
			return;
		}
		cur = cur->next;
	}
}

void SnakeMove(pSnake ps)
{
	pSnakeNode pNext = (pSnakeNode)malloc(sizeof(SnakeNode));
	if (pNext == NULL)
	{
		perror("SnakeMove()::malloc()");
		return;
	}
	pNext->next = NULL;

	switch (ps->dir)
	{
	case UP:
		pNext->x = ps->pSnake->x;
		pNext->y = ps->pSnake->y - 1;
		break;
	case DOWN:
		pNext->x = ps->pSnake->x;
		pNext->y = ps->pSnake->y + 1;
		break;
	case LEFT:
		pNext->x = ps->pSnake->x - 2;
		pNext->y = ps->pSnake->y;
		break;
	case RIGHT:
		pNext->x = ps->pSnake->x + 2;
		pNext->y = ps->pSnake->y;
		break;
	}

	//下一个坐标处是否是食物
	if (NextIsFood(ps, pNext))
	{
		//是食物就吃掉
		EatFood(ps, pNext);
	}
	else
	{
		//不是食物就正常一步
		NotEatFood(ps, pNext);
	}

	//检测撞墙
	KillByWall(ps);

	//检测撞到自己
	KillBySelf(ps);
}


void GameRun(pSnake ps)
{
	

	do
	{
		//当前的分数情况
		SetPos(124, 20);
		printf("总分:%5d\n", ps->score);
		SetPos(124, 22);
		printf("食物的分值:%02d\n", ps->FoodWeight);
		//检测按键
		//上、下、左、右、ESC、空格、F1、F2
		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_ESCAPE))
		{
			ps->status = ESC;
			break;
		}
		else if (KEY_PRESS(VK_SPACE))
		{
			//游戏要暂定
			pause();//暂定和回复暂定
		}
		else if (KEY_PRESS(VK_F1))
		{
			if (ps->SleepTime >= 80)
			{
				ps->SleepTime -= 30;
				ps->FoodWeight += 2;
			}
		}
		else if (KEY_PRESS(VK_F2))
		{
			if (ps->FoodWeight > 2)
			{
				ps->SleepTime += 30;
				ps->FoodWeight -= 2;
			}
		}
		//走一步
		SnakeMove(ps);

		//睡眠一下
		Sleep(ps->SleepTime);

	} while (ps->status == OK);
}



void GameEnd(pSnake ps)
{
	SetPos(30, 24);
	switch (ps->status)
	{
	case ESC:
		printf("主动退出游戏,正常退出\n");
		break;
	case KILL_BY_WALL:
		printf("很遗憾,撞墙了,游戏结束\n");
		break;
	case KILL_BY_SELF:
		printf("很遗憾,咬到自己了,游戏结束\n");
		break;
	}

	pSnakeNode cur = ps->pSnake;

	//释放蛇身的节点
	while (cur)
	{
		pSnakeNode del = cur;
		cur = cur->next;
		free(del);
	}
}








Test.c

#define _CRT_SECURE_NO_WARNINGS 1
#include "Snake.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');
	
}


int main()
{
	//修改当前地区模式,支持中文地区的打印
	setlocale(LC_ALL, "");
	//测试贪吃蛇
	test();
	SetPos(0, 27);//打印程序退出以及代码所在文件地址,这里设置一下是为了美观
	return 0;
}

贪吃蛇视频演示 

贪吃蛇展示

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

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

相关文章

【王道数据结构】【chapter2线性表】【P44t17~t20】【统考真题】

目录 2009年统考 2012年统考 2015年统考 2019年统考 2009年统考 #include <iostream>typedef struct node{int data;node* next; }node,*list;list Init() {list head(list) malloc(sizeof (node));head->next nullptr;head->data-1;return head; }list Buyne…

机器学习 | 如何使用 Seaborn 提升数据分析效率

Seaborn和Matplotlib都是Python可视化库&#xff0c;它们都可以用于创建各种类型的图表。但是&#xff0c;Seaborn 和Matplotlib在概念和设计上有一些不同。 Matplotlib虽然已经是比较优秀的绘图库了&#xff0c;但是它有个今人头疼的问题&#xff0c;那就是API使用过于复杂&am…

人工智能系列:机器的进化(下)

大家好&#xff0c;接着上文的图灵机&#xff0c;继续介绍机器的进化。 1. 第一台计算机 世界上第一台电子计算机是ENIAC&#xff08;埃尼阿克&#xff09;&#xff0c;这是课本上所写的。但计算机学界对于究竟哪台是第一台电子计算机其实存在着争议&#xff0c;除了 ENIAC 以…

[嵌入式软件][启蒙篇][仿真平台] STM32F103实现SPI控制OLED屏幕

上一篇&#xff1a; [嵌入式软件][启蒙篇][仿真平台] STM32F103实现LED、按键 [嵌入式软件][启蒙篇][仿真平台] STM32F103实现串口输出输入、ADC采集 [嵌入式软件][启蒙篇][仿真平台]STM32F103实现定时器 [嵌入式软件][启蒙篇][仿真平台] STM32F103实现IIC控制OLED屏幕 文章目…

【Vue2 + ElementUI】更改el-select的自带的下拉图标为倒三角,并设置相关文字颜色和大小

效果图 实现 <template><div class"search_resources"><div class"search-content"><el-select class"search-select" v-model"query.channel" placeholder"请选择" change"handleResource&q…

【贪吃蛇:C语言实现】

文章目录 前言1.了解Win32API相关知识1.1什么是Win32API1.2设置控制台的大小、名称1.3控制台上的光标1.4 GetStdHandle&#xff08;获得控制台信息&#xff09;1.5 SetConsoleCursorPosition&#xff08;设置光标位置&#xff09;1.6 GetConsoleCursorInfo&#xff08;获得光标…

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;导致网络拥塞…