项目小游戏-贪吃蛇

news2024/11/15 17:36:01

目录

 1.游戏开始  - GameStart

1.1cmd命令窗口

调节窗口命令  

​编辑更改窗口命名 

​编辑

1.2 Win32 API

 win32 API 的介绍: ​编辑

 获取控制台坐标COORD

 获取控制台句柄:

 获取缓冲台光标信息:

获取虚拟键位: 

本地初始化 setlocale();

 游戏开始的具体实现:

 光标隐藏和窗口的设置

 打印环境和功能介绍 

 绘制地图 

创建蛇 

 创建食物 

 游戏运行 - GameRun

 帮助信息的打印

​编辑 

 检测按键的情况操控蛇的身体

 判断蛇下一步走向是食物还是没有食物

 判断蛇每走一步状态是否正常

结束游戏 - 善后工作 - GameEnd();

实现代码参考:

 头文件 - > Snack.h

 源文件  - > Snack.c

 测试文件 - > text.c


前言:本次给大家带来的是贪吃蛇小游戏项目的实现!项目实现中会涉及C语言语法,WIn32 API ,

链表,本地化设置等等,下面就跟着我一起来实现吧!

 游戏实现效果呈现:

贪吃蛇游戏展现

 游戏流程设计:

本次游戏程序的实现分为3个部分,游戏开始;游戏运行;游戏结束,下面大家就跟着我来一个一个实现吧! 

 1.游戏开始  - GameStart

首先我们来认识几个知识点:

  1. 为了实现我们贪吃蛇游戏的运行界面,我们使用windows电脑的cmd命令窗口来展现我们贪吃蛇游戏的界面运行.
  2. 在运行贪吃蛇的过程中我们需要打印欢迎界面,功能界面,以及贪吃蛇的地图,贪吃蛇,获取操控贪吃蛇的按键状态等等
  3. 为了打印地图和贪吃蛇的外观,食物我们需要用到宽字符,需要对编译器本地化.

1.1cmd命令窗口

调节窗口命令  

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

为了更加方便的游玩游戏,我们需要一个合适大小的窗口,以上的代码命令可以更改cmd窗口的大小,

cols代表x轴的长度,lines代表的是y轴的长度.

注意:cmd命令窗口的x轴和y轴和原本的坐标系不同,下图作为解析: 

更改窗口命名 

system("title 贪吃蛇");

 title 后面的名称可以是中文也可以是英文.

以上两个命令准备使用的都是system函数,以下是更多信息介绍: 

1.2 Win32 API

 win32 API 的介绍: 

 获取控制台坐标COORD

我们前面了解了命令窗口cmd的坐标关系,那么怎么获得每个点位具体的坐标呢,我们就需要使用到win32 API 中的COORD结构体.

COORD: 

获取坐标: 

// 获取控制台坐标
COODR pos = {x, y};
//pos 是我们自定义的变量
//x, y 分别是坐标轴上的x和y坐标
 获取控制台句柄:

我们知道要操控一个东西,需要一定的物品或者条件,如我们要开车,那么方向盘就必不可少,那么我们要操作命令行cmd窗口又需要什么呢?答案是句柄!

HANDLE hOutput = GetStdhandle(STD_OUTPUT_HANDLE);
 获取缓冲台光标信息:

我们每次使用cmd窗口时,会有光标闪烁提示我们打印的方位接下来将从什么地方开始,但是我们运行贪吃蛇游戏时不希望出现光标,那么我们需要对光标进行操作将其隐藏起来! 

注:光获取光标信息还不够,我们要在获取后更改信息再设置一遍,就好比你打游戏时更改键位要按保存一样! 

设置定位光标位置: 

在贪吃蛇游戏中我们的贪吃蛇是时刻走动的,地图的范围也是固定,周围的提示信息也需要在合适的位置打印,这些都离不开定位光标的位置,因此我们需要一个可以定位光标位置的封装函数,结合前面所介绍的,我们就可以进行实现了 

// 定位光标位置
void SetPos(short x, short y)
{
    // 获取句柄
    HANDLE hOutput = GetStdHandle(STD_OUTPUT_HANDLE);
    // 定位坐标
    COORD pos = {x, y};
    // 定位光标位置
	SetConsoleCursorPosition(houtput, pos);

}
获取虚拟键位: 

我们知道贪吃蛇游戏中需要玩家使用键盘上的上下左右对贪吃蛇进行操作,那么计算机又是如何判断我们是否按下了相应的键位呢?这就离不开获取虚拟键位的函数:GetAsyncKeyState

注:为了方便使用我们将其定义为宏. 

 键位的虚拟值定义查询:GetAsyncKeyState 函数 (winuser.h) - Win32 应用 |Microsoft学习

本地初始化 setlocale();

在贪吃蛇游戏的实现中,我们需要打印一些宽字符作为地图的围墙 -> □ ,还有贪吃蛇的身体 -> ●,食物的形状 -> ★ ◆ ▲等等,但是这些字符在C语言底层的并不存在,因为C的来源是美国,其他国家后来对应的字符,语言都是经过后续添加进入的,所以我们要使用宽字符就需要先切换为本地模式(开始固定为C模式)。

// 设置适配本地环境
setlocale(LC_ALL, "");

 游戏开始的具体实现:

了解以上的知识后我们就可以开始实现我们的游戏开始模块的代码了,分为以下几个步骤: 

  •          0.光标隐藏和窗口的设置
  •         1.打印环境和功能介绍 
  •         2.绘制地图 
  •         3.创建蛇 
  •         4.创建食物 

 光标隐藏和窗口的设置

//窗口的设置
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);

注意:小编是函数封装的,单独测试的小伙伴记得使用main函数,并且包含头文件 :

#include <Windows.h>

设置完毕后运行的状况以下:  

 打印环境和功能介绍 

展示如下:  

// 欢迎界面的打印
void WelcomeToGame()
{
	SetPos(35, 14);
	wprintf(L"欢迎来到贪吃蛇小游戏\n");
	SetPos(38, 20);
	system("pause");
	system("cls");
	SetPos(28, 14);
	wprintf(L"用 ↑. ↓. ←. → 来控制蛇的移动,按F3加速,F4减速\n");
	SetPos(28, 15);
	wprintf(L"加速能够得到更高的分数\n");
	SetPos(38, 20);
	system("pause");
	system("cls");
}

代码中的SetPos(); 函数我们在前面已经介绍,用于定位坐标。

// wprintf 的使用方法和printf相同, 只是这边用于中文打印两者都可以打印,这边提前使用是为了与后面宽字符打印统一

printf("您好! CSDN");
wprintf(L"您好! CSDN");

 system中的pause用于暂停 -> 防止程序直接结束,cls用于清屏。

 绘制地图 

在使用代码打印地图之前,我们需要考虑好要绘制多大的地图,并且我们要知道蛇的身体和蛇的食物是宽字符组成,需要占两个字节的位置,所以我们地图的内部需要是偶数成对的方块,不然会出现蛇的身体一半在墙内一半墙外,食物一半墙内一半墙外的情况。 

以下是我的地图设计: 

 有了设计图纸后我们创建打印方面就很简单了,分为上下左右四个区域进行打印,要注意打印的个数和打印的坐标(使用SetPos函数调节)。

#define WALL L'□'
// 3.绘制地图 
void CreateMap()
{
	// 上
	int i = 0;
	for (i = 0; i < 29; i++)
	{
		wprintf(L"%lc", WALL);
	}

	// 下
	SetPos(0, 26);
	for (i = 0; i < 29; i++)
	{
		wprintf(L"%lc", WALL);
	}

	// 左
	for (i = 1; i <= 25; i++)
	{
		SetPos(0, i);
		wprintf(L"%lc", WALL);
	}

	// 右
	for (i = 1; i <= 25; i++)
	{
		SetPos(56, i);
		wprintf(L"%lc", WALL);
	}

}

创建蛇 

地图绘制完毕后,我们就需要考虑贪吃蛇的创建了,并且我们需要将贪吃蛇打印在地图中。

 首先我们创建的贪吃蛇需要考虑以下几个点:

1.贪吃蛇的身体应由链表组成,所以我们需要创建贪吃蛇的身体节点,并且存放坐标以及指向下一截身体的指针。 

// 创建贪吃蛇身的节点
typedef struct SnackNode
{
	// 坐标
	int x;
	int y;
	// 指向下一个节点的指针
	struct SnackNode* next;
}SnackNode, * pSnackNode;

2.我们需要创建面向对象:贪吃蛇 

贪吃蛇对象应该包含以下几点:

  1. 指向贪吃蛇头的结构体指针;用来判断贪吃蛇的方向。
  2. 指向贪吃蛇所吃食物的结构体指针;方便贪吃蛇吃下食物。 
  3. 贪吃蛇的方向;贪吃蛇是向上向下还是左或者右。
  4. 贪吃蛇的状态;贪吃蛇是否是正常运行,撞墙死亡,撞自己死亡,正常退出游戏。
  5. 总分数的显示;
  6. 每个食物的分数;
  7. 睡眠时间;要实现贪吃蛇移动需要让我们的视觉看到一会打印蛇身一会消失向前打印,这就需要使用睡眠程序实现。
// 面向对象:贪吃蛇
typedef struct Snake
{
	pSnackNode _pSnake;  // 指向蛇头的指针
	pSnackNode _pFood;   // 指向食物节点的指针
	enum DIRECTION _dir; // 蛇的方向
	enum GAME_STATUS _status; //蛇的状态
	int _food_weight; // 一个食物的分数
	int _score;		// 总分数
	int _sleep_time; // 休息时间,时间越短,速度越快,时间越长,速度越慢
}Snake, * pSnake;

以下是对象蛇的类型声明: 

// 蛇的状态
// 正常, 撞墙, 撞到自己, 正常退出
enum GAME_STATUS
{
	OK,				//正常
	KILL_BY_WALL,	//撞墙
	KILL_BY_SELF,	//撞到自己
	END_NORMAL		//正常退出
};

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

 初始化蛇:

创建好后我们需要初始化蛇,其中包括蛇的身体连接,蛇身初始的打印位置,蛇的方向,蛇的状态,总分数,食物分数,打印蛇身等等。 

#define POS_X 24
#define POS_y 5
#define BODY L'●'
#define FOOD L'★'
#define FOOD1 L'◆'
#define FOOD2 L'▲'

// 4.创建蛇 
void InitSnack(pSnake ps)
{
	int i = 0;

	pSnackNode cur = NULL;

	// 创建五个节点(蛇身)

	for (i = 0; i < 5; i++)
	{
		cur = (pSnackNode)malloc(sizeof(SnackNode));

        // 申请失败
		if (cur == NULL)
		{
			perror("InitSnack()::malloc()");
			return;
		}

        // 申请成功
		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;
		}
	}

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

	// 设置贪吃蛇的属性
	ps->_dir = RIGHT; // 默认向右
	ps->_score = 0;		//总分数
	ps->_food_weight = 10; // 食物的分数
	ps->_sleep_time = 200;// 单位毫秒
	ps->_status = OK; // 蛇的状态

}

 创建食物 

 首先在创建食物之前我们需要明确,食物的生成是随机的,在地图内的,生成的坐标要是倍数坐标,这样蛇才可以吃到,随机方面我们就需要使用到随机函数rank,但是系统的随机是伪随机,要做到真正的随机我们还需要使用到时间戳,并且设定一下。生成的过程中我们不可以与蛇的身体冲突,最后进行打印,然后传入面向对象贪吃蛇统一管理。

// 设定随机rank
srand((unsigned int)time(NULL));
// 5.创建食物 
void CreateFood(pSnake ps)
{
	// 坐标
	int x = 0;
	int y = 0;

	// 生成x是2的倍数
	// x: 2-54
	// y: 1-25

again:
	do
	{
		x = rand() % 53 + 2;
		y = rand() % 25 + 1;
	} while (x % 2 != 0);

	// x和y的坐标不可以和蛇冲突
	pSnackNode cur = ps->_pSnake;
	while (cur)
	{
		if (x == cur->x && y == cur->y)
		{
            // 如果冲突重新生成食物
			goto again;
		}
		cur = cur->next;
	}

	// 创建食物节点
	pSnackNode pFood = (pSnackNode)malloc(sizeof(SnackNode));

    // 创建失败
	if (pFood == NULL)
	{
		perror("CreateFood()::malloc()");
		return;
	}

    // 创建成功
	pFood->x = x;
	pFood->y = y;
	pFood->next = NULL;


	SetPos(x, y);//定位打印

    // 根据不同分数生成不同形状的食物
 
	if (ps->_score <= 50)
	{
		wprintf(L"%lc", FOOD);
	}
	else if (ps->_score > 50 && ps->_score <= 100)
	{
		wprintf(L"%lc", FOOD1);
		ps->_food_weight += 2;
	}
	else if (ps->_score > 100)
	{
		wprintf(L"%lc", FOOD2);
		ps->_food_weight += 4;
	}
    
    // 将设定好的食物信息传入面向对象贪吃蛇中
	ps->_pFood = pFood;
}

最后效果:  

 游戏运行 - GameRun

 这一个环节主要实现以下内容:

  1. 帮助信息的打印
  2. 检测按键的情况操控蛇的身体
  3. 判断蛇下一步走向是食物还是没有食物
  4. 判断蛇每走一步状态是否正常

 帮助信息的打印

//打印帮助信息
void PrintHelpInfo()
{
	SetPos(64, 14);
	wprintf(L"%ls", L"不能穿墙,不能咬到自己");
	SetPos(64, 15);
	wprintf(L"%ls", L"用 ↑. ↓. ←. → 来控制蛇的移动");
	SetPos(64, 16);
	wprintf(L"%ls", L",按F3加速,F4减速");
	SetPos(64, 17);
	wprintf(L"%ls", L"按ESC退出游戏,按空格暂停游戏");
	SetPos(64, 18);
	wprintf(L"%ls", L"@阳区欠出品");
	
}

// 打印总分数和食物的分数
SetPos(64, 10);
printf("总分数:%04d", ps->_score);
SetPos(64, 11);
printf("当前食物的分数:%02d", ps->_food_weight);
SetPos(64, 12);
printf("当前蛇的速度:%4d(毫秒)", ps->_sleep_time);

实现界面如下: 

 

 检测按键的情况操控蛇的身体

 前面我们提到检测按键的Win32 API 中包含的函数,我们可以先封装一个宏。

#define KEY_PRESS(VK) ( (GetAsyncKeyState(VK) & 0x1) ? 1 : 0 )
// 检测按键
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))// ESC退出游戏
{
	// 正常退出游戏
	ps->_status = END_NORMAL;
}
else if (KEY_PRESS(VK_F3))// F3加速
{
	// 加速 -> 为了防止无限加速设定条件
	if (ps->_sleep_time > 100)
	{
		ps->_sleep_time -= 30;
		ps->_food_weight += 2;
	}
}
else if (KEY_PRESS(VK_F4))// F4减速
{
	// 减速  -> 为了防止无限减速设定条件
	if (ps->_sleep_time >= 80 && ps->_sleep_time <= 300)
	{
		ps->_sleep_time += 30;
		ps->_food_weight -= 2;
	}
}

暂停实现: 

暂停的底层逻辑就是一直休眠程序,知道特定的条件解除。 

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

 判断蛇下一步走向是食物还是没有食物

// 判断下一个坐标是否是食物
int NextIsFood(pSnackNode pn, pSnake ps)
{
	return (ps->_pFood->x == pn->x && ps->_pFood->y == pn->y);
}

// 下一个是食物,吃掉
void EatFood(pSnackNode pn, pSnake ps)
{
	// 头插法
	ps->_pFood->next = ps->_pSnake;
	ps->_pSnake = ps->_pFood;

	// 释放下一个位置的节点
	free(pn);
	pn = NULL;

	// 打印
	pSnackNode cur = ps->_pSnake;
	while (cur)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%lc", BODY);
		cur = cur->next;
	}
	ps->_score += ps->_food_weight;
	ps->_anger_num += 1;
	// 重新创建食物
	CreateFood(ps);
}

// 下一个位置不是食物
void NoFood(pSnackNode pn, pSnake ps)
{
	// 头插法
	pn->next = ps->_pSnake;
	ps->_pSnake = pn;

	pSnackNode cur = ps->_pSnake;

//  找到最后一个节点的之前节点
	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("  ");

	// 释放最后一个节点
	free(cur->next);
	// 把倒数第二个节点的地址置为NULL
	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)
{
	pSnackNode cur = ps->_pSnake->next;
	while (cur)
	{
		if (cur->x == ps->_pSnake->x && cur->y == ps->_pSnake->y)
		{
			ps->_status = KILL_BY_SELF;
		}
		cur = cur->next;
	}
}

 以下是贪吃蛇走动的实现:

//贪吃蛇走一步->蛇的移动
void SnakeMove(pSnake ps)
{
	pSnackNode pNextNode = (pSnackNode)malloc(sizeof(SnackNode));
	if (pNextNode == NULL)
	{
		perror("SnakeMove()::malloc()");
		return;
	}

    // 注意:以下的x轴坐标必须是偶数倍的加减
	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);
}

结束游戏 - 善后工作 - GameEnd();

我们需要知道游戏内的蛇身是通过空间动态内存申请的,再结束后我们都需要释放掉,并且我们需要依据蛇的不同状态反馈玩家游戏结束的原因。 

// 结束游戏 - 善后工作
void GameEnd(pSnake ps)
{
	SetPos(24, 12);
	switch (ps->_status)
	{
	case END_NORMAL:
		wprintf(L"你主动结束游戏\n");
		break;
	case KILL_BY_WALL:
		wprintf(L"你撞到了墙上,游戏结束了\n");
		break;
	case KILL_BY_SELF:
		wprintf(L"你撞到了自己,游戏结束了\n");
		break;
	}
	SetPos(0, 26);

	// 释放蛇的身体
	pSnackNode cur = ps->_pSnake;
	while (cur)
	{
		pSnackNode del = cur;
		cur = cur->next;
		free(del);
	}

}

实现代码参考:

  1.  头文件 - > Snack.h
  2. 源文件  - > Snack.c
  3. 测试文件 - > text.c

 头文件 - > Snack.h

#pragma once
#define _CRT_SECURE_NO_WARNINGS 1
#include <Windows.h>
#include <stdbool.h>
#include <stdio.h>
#include <time.h>

#define POS_X 24
#define POS_y 5
#define BODY L'●'
#define WALL L'□'
#define FOOD L'★'
#define FOOD1 L'◆'
#define FOOD2 L'▲'

// 类型的声明

// 蛇的状态
// 正常, 撞墙, 撞到自己, 正常退出
enum GAME_STATUS
{
	OK,				//正常
	KILL_BY_WALL,	//撞墙
	KILL_BY_SELF,	//撞到自己
	END_NORMAL		//正常退出
};

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

// 创建贪吃蛇身的节点
typedef struct SnackNode
{
	// 坐标
	int x;
	int y;
	// 指向下一个节点的指针
	struct SnackNode* next;
}SnackNode, * pSnackNode;

// 面向对象:贪吃蛇
typedef struct Snake
{
	pSnackNode _pSnake;  // 指向蛇头的指针
	pSnackNode _pFood;   // 指向食物节点的指针
	enum DIRECTION _dir; // 蛇的方向
	enum GAME_STATUS _status; //蛇的状态
	int _food_weight; // 一个食物的分数
	int _score;		// 总分数
	int _sleep_time; // 休息时间,时间越短,速度越快,时间越长,速度越慢
	int _anger_num; // 蛇的怒气
}Snake, * pSnake;

// 函数的声明

// 初始化游戏
void GameStart(pSnake ps);

// 欢迎界面的打印
void WelcomeToGame();

//定位光标位置
void setpos(short x, short y);

// 3.绘制地图 
void CreateMap();

// 4.初始化蛇
void InitSnack(pSnake ps);

// 5.创建食物 
void CreateFood(pSnake ps);

// 运行游戏的逻辑
void GameRun(pSnake ps);

//贪吃蛇走一步->蛇的移动
void SnakeMove(pSnake ps);

// 判断下一个坐标是否是食物
int NextIsFood(pSnackNode pn, pSnake ps);

// 下一个是食物,吃掉
void EatFood(pSnackNode pn, pSnake ps);

// 下一个位置不是食物
void NoFood(pSnackNode pn, pSnake ps);

// 检测蛇是否撞墙
void KillByWall(pSnake ps);

// 检测蛇是否撞到自己
void KillBySelf(pSnake ps);

// 结束游戏 - 善后工作
void GameEnd(pSnake ps);

 源文件  - > Snack.c

#define _CRT_SECURE_NO_WARNINGS 1
#include "snack_game.h"

//定位光标位置
void SetPos(short x, short y)
{
	// 获得坐标
	COORD pos = { x, y };
	// 获取标准输出设备的句柄
	HANDLE houtput = GetStdHandle(STD_OUTPUT_HANDLE);
	// 定位光标位置
	SetConsoleCursorPosition(houtput, pos);
}

// 欢迎界面的打印
void WelcomeToGame()
{
	SetPos(35, 14);
	wprintf(L"欢迎来到贪吃蛇小游戏\n");
	SetPos(38, 20);
	system("pause");
	system("cls");
	SetPos(28, 14);
	wprintf(L"用 ↑. ↓. ←. → 来控制蛇的移动,按F3加速,F4减速\n");
	SetPos(28, 15);
	wprintf(L"加速能够得到更高的分数\n");
	SetPos(38, 20);
	system("pause");
	system("cls");
}

// 3.绘制地图 
void CreateMap()
{
	// 上
	int i = 0;
	for (i = 0; i < 29; i++)
	{
		wprintf(L"%lc", WALL);
	}

	// 下
	SetPos(0, 26);
	for (i = 0; i < 29; i++)
	{
		wprintf(L"%lc", WALL);
	}

	// 左
	for (i = 1; i <= 25; i++)
	{
		SetPos(0, i);
		wprintf(L"%lc", WALL);
	}

	// 右
	for (i = 1; i <= 25; i++)
	{
		SetPos(56, i);
		wprintf(L"%lc", WALL);
	}

}

// 4.创建蛇 
void InitSnack(pSnake ps)
{
	int i = 0;

	pSnackNode cur = NULL;
	// 创建五个节点
	for (i = 0; i < 5; i++)
	{
		cur = (pSnackNode)malloc(sizeof(SnackNode));
		if (cur == NULL)
		{
			perror("InitSnack()::malloc()");
			return;
		}
		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;
		}
	}

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

	// 设置贪吃蛇的属性
	ps->_dir = RIGHT; // 默认向右
	ps->_score = 0;		//总分数
	ps->_food_weight = 10; // 食物的分数
	ps->_sleep_time = 200;// 单位毫秒
	ps->_status = OK; // 蛇的状态

}

// 5.创建食物 
void CreateFood(pSnake ps)
{
	// 坐标
	int x = 0;
	int y = 0;

	// 生成x是2的倍数
	// x: 2-54
	// y: 1-25
again:
	do
	{
		x = rand() % 53 + 2;
		y = rand() % 25 + 1;
	} while (x % 2 != 0);

	// x和y的坐标不可以和蛇冲突
	pSnackNode cur = ps->_pSnake;
	while (cur)
	{
		if (x == cur->x && y == cur->y)
		{
			goto again;
		}
		cur = cur->next;
	}

	// 创建食物节点
	pSnackNode pFood = (pSnackNode)malloc(sizeof(SnackNode));
	if (pFood == NULL)
	{
		perror("CreateFood()::malloc()");
		return;
	}
	pFood->x = x;
	pFood->y = y;
	pFood->next = NULL;

	SetPos(x, y);//定位
	if (ps->_score <= 50)
	{
		wprintf(L"%lc", FOOD);
	}
	else if (ps->_score > 50 && ps->_score <= 100)
	{
		wprintf(L"%lc", FOOD1);
		ps->_food_weight += 2;
	}
	else if (ps->_score > 100)
	{
		wprintf(L"%lc", FOOD2);
		ps->_food_weight += 4;
	}

	ps->_pFood = pFood;
}

// 初始化游戏
void GameStart(pSnake ps)
{

// 0.光标隐藏和窗口的设置
	//窗口的设置
	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);

// 1.打印环境界面和功能介绍
	WelcomeToGame(); 
// 3.绘制地图 
	CreateMap();
// 4.创建蛇 
	InitSnack(ps);
// 5.创建食物 
	CreateFood(ps);
}

//打印帮助信息
void PrintHelpInfo()
{
	SetPos(64, 14);
	wprintf(L"%ls", L"不能穿墙,不能咬到自己");
	SetPos(64, 15);
	wprintf(L"%ls", L"用 ↑. ↓. ←. → 来控制蛇的移动");
	SetPos(64, 16);
	wprintf(L"%ls", L",按F3加速,F4减速");
	SetPos(64, 17);
	wprintf(L"%ls", L"按ESC退出游戏,按空格暂停游戏");
	SetPos(64, 18);
	wprintf(L"%ls", L"@阳区欠出品");
	
}

#define KEY_PRESS(VK) ( (GetAsyncKeyState(VK) & 0x1) ? 1 : 0 )
void Pause()
{
	while (1)
	{
		Sleep(200);
		if (KEY_PRESS(VK_SPACE))
		{
			break;
		}
	}
}

// 判断下一个坐标是否是食物
int NextIsFood(pSnackNode pn, pSnake ps)
{
	return (ps->_pFood->x == pn->x && ps->_pFood->y == pn->y);
}

// 下一个是食物,吃掉
void EatFood(pSnackNode pn, pSnake ps)
{
	// 头插法
	ps->_pFood->next = ps->_pSnake;
	ps->_pSnake = ps->_pFood;

	// 释放下一个位置的节点
	free(pn);
	pn = NULL;

	// 打印
	pSnackNode cur = ps->_pSnake;
	while (cur)
	{
		SetPos(cur->x, cur->y);
		wprintf(L"%lc", BODY);
		cur = cur->next;
	}
	ps->_score += ps->_food_weight;
	ps->_anger_num += 1;
	// 重新创建食物
	CreateFood(ps);
}

// 下一个位置不是食物
void NoFood(pSnackNode pn, pSnake ps)
{
	// 头插法
	pn->next = ps->_pSnake;
	ps->_pSnake = pn;

	pSnackNode cur = ps->_pSnake;
	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("  ");

	// 释放最后一个节点
	free(cur->next);
	// 把倒数第二个节点的地址置为NULL
	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)
{
	pSnackNode cur = ps->_pSnake->next;
	while (cur)
	{
		if (cur->x == ps->_pSnake->x && cur->y == ps->_pSnake->y)
		{
			ps->_status = KILL_BY_SELF;
		}
		cur = cur->next;
	}
}


//贪吃蛇走一步->蛇的移动
void SnakeMove(pSnake ps)
{
	pSnackNode pNextNode = (pSnackNode)malloc(sizeof(SnackNode));
	if (pNextNode == NULL)
	{
		perror("SnakeMove()::malloc()");
		return;
	}
	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);
}

// 运行游戏
void GameRun(pSnake ps)
{
	//打印帮助信息
	PrintHelpInfo();
	do
	{
		// 打印总分数和食物的分数
		SetPos(64, 10);
		printf("总分数:%04d", ps->_score);
		SetPos(64, 11);
		printf("当前食物的分数:%02d", ps->_food_weight);
		SetPos(64, 12);
		printf("当前蛇的速度:%4d(毫秒)", ps->_sleep_time);

		// 检测按键
		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_NORMAL;
		}
		else if (KEY_PRESS(VK_F3))
		{
			// 加速
			if (ps->_sleep_time > 100)
			{
				ps->_sleep_time -= 30;
				ps->_food_weight += 2;
			}
		}
		else if (KEY_PRESS(VK_F4))
		{
			// 减速
			if (ps->_sleep_time >= 80 && ps->_sleep_time <= 300)
			{
				ps->_sleep_time += 30;
				ps->_food_weight -= 2;
			}
		}

		//贪吃蛇走一步
		SnakeMove(ps);

		Sleep(ps->_sleep_time);

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

// 结束游戏 - 善后工作
void GameEnd(pSnake ps)
{
	SetPos(24, 12);
	switch (ps->_status)
	{
	case END_NORMAL:
		wprintf(L"你主动结束游戏\n");
		break;
	case KILL_BY_WALL:
		wprintf(L"你撞到了墙上,游戏结束了\n");
		break;
	case KILL_BY_SELF:
		wprintf(L"你撞到了自己,游戏结束了\n");
		break;
	}
	SetPos(0, 26);

	// 释放蛇的身体
	pSnackNode cur = ps->_pSnake;
	while (cur)
	{
		pSnackNode del = cur;
		cur = cur->next;
		free(del);
	}

}

 测试文件 - > text.c

#define _CRT_SECURE_NO_WARNINGS 1
#include "snack_game.h"
#include <locale.h>


// 游戏测试的逻辑
void text()
{
	int ch = 0;
	do
	{
		system("cls");
		// 创建贪吃蛇
		Snake snake = { 0 };
		// 初始化游戏
		// 0.光标隐藏和窗口的设置
		// 1.打印环境 
		// 2.功能介绍 
		// 3.绘制地图 
		// 4.创建蛇 
		// 5.创建食物 
		// 6.设置游戏相关信息
		GameStart(&snake);

		// 运行游戏
		GameRun(&snake);

		// 结束游戏 - 善后工作
		GameEnd(&snake);
		SetPos(20, 15);
		printf("再来一局吗?(Y/N):");
		ch = getchar();
		while (getchar() != '\n');

	} while (ch == 'Y' || ch == 'y');
	SetPos(0, 27);
	system("cls");

}

int main()
{
	// 设置适配本地环境
	setlocale(LC_ALL, "");
	srand((unsigned int)time(NULL));
	text();
	return 0;
}

 

 

 

 

 

 

 

 

 

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

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

相关文章

uniapp中scroll-view初始化的时候 无法横向滚动到某个为止

项目需求 实现日历&#xff08;13天&#xff09;默认高亮第六天 并定位到第六 左边右边各六天&#xff08;可以滑动&#xff09; 直接上代码 <template><scroll-view class"scroll-X":show-scrollbar"true" :scroll-x"scrollable":…

理解这几个安全漏洞,你也能做安全测试!

如今安全问题显得越来越重要&#xff0c;一个大型的互联网站点&#xff0c;你如果每天查看日志&#xff0c;会发现有很多尝试攻击性的脚本。 如果没有&#xff0c;证明网站影响力还不够大。信息一体化的背后深藏着各类安全隐患&#xff0c;例如由于开发人员的不严谨导致为Web应…

八字入门书介绍

《千里命稿》&#xff0c;此书是民国时期上海的韦千里先生所著&#xff0c;是他主讲的命学培训班的讲义&#xff0c;成书时间在1935年。该书语言简练、述理清晰&#xff0c;是讲述子平命学概念的佼佼者&#xff0c;尤其是对五行、六神的性质讲解&#xff0c;可以说是此书的精华…

xpath的使用以及原理-元素定位

# 查找文本框输入文本 driver.find_element(By.CLASS_NAME,"nav-search-input").send_keys("i_cecream查找到了") #查找到之后点击 driver.find_element(By.CLASS_NAME,"nav-search-btn").click()time.sleep(30)selenium4的解析。 client调用se…

Elasticsearch进阶篇(三):ik分词器的使用与项目应用

ik分词器的使用 一、下载并安装1.1 已有作者编译后的包文件1.2 只有源代码的版本1.3 安装ik分词插件 二、ik分词器的模式2.1 ik_smart演示2.2 ik_max_word演示2.3 standard演示 三、ik分词器在项目中的使用四、ik配置文件4.1 配置文件的说明4.2 自定义词库 五、参考链接 一、下…

动态内存的管理

1.为什么要有动态数组呢 int val 20;//在栈空间上开辟四个字节 char arr[10] {0};//在栈空间上开辟10个字节的连续空间 上述开辟空间的方式有两个特点&#xff1a; 空间开辟大小是固定的数组在申明的时候&#xff0c;必须指定数组的长度&#xff0c;数组空间一旦确定了大小…

【AI自媒体制作】【AI工具】天工AI

链接&#xff1a;天工AI 目前chatgpt3.5已经免费&#xff0c;很多AI平台都可以进行一定层度的白嫖&#xff0c;对于个人开发者是一件好事&#xff0c;有些会员就没必要充值了。 天工AI是比较常见的AI工具了&#xff0c;可以识别图片、智能问答、生成图片等。当然对于一个程序员…

关于电商独立站搭建中电商API数据采集接口的应用

搭建供应链系统时&#xff0c;您可能需要与电商平台进行集成&#xff0c;以实现订单管理、库存同步、物流跟踪等功能。以下是一些常见的电商接口&#xff0c;可以帮助您构建供应链系统&#xff1a; 1. **淘宝开放平台接口**&#xff1a;淘宝开放平台提供了丰富的接口&#xff…

上位机图像处理和嵌入式模块部署(树莓派4b和驱动的编写)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 树莓派4b上面还支持驱动代码的编写&#xff0c;这是我没有想到的。这里驱动&#xff0c;更多的是一种框架的编写&#xff0c;不一定是编写真正的驱…

Selenium web自动化测试环境搭建

Selenium web自动化环境搭建主要要经历以下几个步骤&#xff1a; 1、安装python 在python官网&#xff1a;Welcome to Python.org&#xff0c;根据各自对应平台如&#xff1a;windows&#xff0c;下载相应的python版本。 ​ 下载成功后&#xff0c;点击安装包&#xff0c;一直…

排序之插入排序:从斗地主到插入排序

目录 1.斗地主如何摸牌 2.从摸牌想到插入排序 3.完成插入排序 4.结束语 1.斗地主如何摸牌 不知道各位是否玩过几乎人人都玩过的斗地主游戏呢&#xff1f;相必各位或多或少都玩过一点&#xff0c;再没玩过也看别人打过。今天博主就将从这个游戏为大家讲解我们的插入排序。 在…

shell编写

运算符 echo 打印 unset 删除 [ boolean表达式 ] 注意中括号中有两个空格&#xff0c;两个空格中添加表达式 0为true 1为false $?查看结果 -ge 大于等于 lt小于 le小于等于 eq等于 ne不等于 格式& | [ 表达式 ] -o是或者or的意思 -a是and的意思 &#xff01…

访问学者申请的成功经验

在申请成为访问学者时&#xff0c;经验是至关重要的。下面知识人网小编将介绍一些可以帮助您成功申请的经验和技巧。 首先&#xff0c;了解目标机构或大学的研究方向和需求是非常重要的。在申请之前&#xff0c;仔细研究该机构的学术项目、研究成果以及教授的专业领域&#xff…

kaggle 泰坦尼克号2 得分0.7799

流程 导入所要使用的包引入kaggle的数据集csv文件查看数据集有无空值填充这些空值提取特征分离训练集和测试集调用模型 导入需要的包 import pandas as pd import numpy as np import matplotlib.pyplot as plt import seaborn as sns import warnings warnings.filterwarni…

Redis 逻辑过期策略设计思路

引言&#xff1a; 当我们平常使用Redis缓存的时候&#xff0c;会出现一种场景&#xff0c; redis的key到过期时间了&#xff0c;总是需要到数据库里面去查一遍数据再set回redis&#xff0c;这个时候如果数据库响应比较慢&#xff0c;那么就会造成用户等待&#xff0c;如果刚好…

python爬虫小案例——汽车之家

本篇文章是使用bs4中的BeautifulSoup和requests解析网页和获取数据&#x1f451;&#x1f31f; 文章目录 &#x1f31f;前言一、&#x1f349;bs4中的BeautifulSoup二、&#x1f349;bs4的语法三、&#x1f349;内容实践1. 确定想要爬取的内容2. 分析网页3. 获取数据分析 &…

【ThinkPHP框架教程·Part-02】开发规范和目录结构

文章目录 一、开发规范1、目录和文件的规范2、函数和类、属性命名规范3、常量与配置的规范4、数据表和字段的规范 二、目录结构1、单应用模式目录结构图&#xff08;默认&#xff09;2、多应用模式目录结构图&#xff08;自定义&#xff09;3、仅允许public目录对外访问4、其他…

软考141-上午题-【软件工程】-杂题+小结

一、杂题 真题1&#xff1a; 真题2&#xff1a; 真题3&#xff1a; 真题4&#xff1a; 真题5&#xff1a; 真题6&#xff1a; 真题7&#xff1a; 真题8&#xff1a; 真题9&#xff1a; 真题10&#xff1a; 真题11&#xff1a; 真题12&#xff1a; 真题13&#xff1a; 真题14&a…

4.20 IO流

IO流结构 InputStream&#xff08;字节输入流&#xff09; public static void main(String[] args) {// byteInputStream();// byteInputStream1();// byteInputStream2();byteInputStream3();}// 使用字节流时对于中文汉字基本都会出现乱码问题&#xff0c;因此对中文乱码问…

ccfcsp201312-2 ISBN号码

注意&#xff1a;50分 -- u10&#xff0c;最后一位为X 代码&#xff1a; #include <bits/stdc.h> using namespace std; string s; int a[12]; int main() {cin >> s;a[1] s[0] - 0;a[2] s[2] - 0;a[3] s[3] - 0;a[4] s[4] - 0;a[5] s[6] - 0;a[6] s[7] - …