C++ vector容器实现贪吃蛇小游戏

所属分类: 软件编程 / C 语言 阅读数: 114
收藏 0 赞 0 分享

本文实例为大家分享了C++ vector容器 实现贪吃蛇,供大家参考,具体内容如下

使用vector容器实现贪吃蛇简化了很多繁琐操作,且相比之前我的代码已经做到了尽量的简洁

技术环节:

编译环境:windows VS2019

需求:

控制贪吃蛇吃食物,吃到一个食物蛇身变长一节,得分增加,撞墙或撞自己则游戏结束。

思路:

创建一个vector容器,容器内存储蛇的每节身体的结构变量,结构变量中保存蛇身体的xy坐标,通过使用vector成员方法不断添加和删除容器中的数据,实现蛇坐标的规律移动,吃到食物等时执行对应操作。

在代码注释中标注了每一步是怎么实现的。

注意:

由于编译器原因程序中_kbhit()和_getch()函数可能在其他编译器上编译会出现错误,解决办法是去掉函数前面的“_”。

运行效果:

#include <iostream>
#include <vector>
#include <windows.h>
#include <conio.h>
#include <ctime>

using namespace std;


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

//食物类
class Food
{
private:
 int m_x;
 int m_y;
public:
 void randfood() //随机产生一个食物
 {
 srand((int)time(NULL));
 L1:
 m_x = rand() % (85) + 2;
 m_y = rand() % (25) + 2;

 if (m_x % 2) //如果食物的x坐标不是偶数则重新确定食物的坐标
 goto L1;

 gotoxy(m_x, m_y); //在确认好的位置输出食物
 cout << "★";
 }
 int getFoodm_x() //返回食物的x坐标
 {
 return m_x;
 }
 int getFoodm_y() //返回食物的y坐标
 {
 return m_y;
 }
};

//蛇类
class Snake
{
private:
 //蛇坐标结构
 struct Snakecoor
 {
 int x;
 int y;
 };
 //蛇容器
 vector<Snakecoor> snakecoor;


 //判断和改变方向函数
 void degdir(Snakecoor& nexthead) //参数:新蛇头结构变量、蛇坐标容器
 {
 static char key = 'd'; //静态变量防止改变移动方向后重新改回来

 if (_kbhit()) //改变蛇前进的方向
 {
 char temp = _getch();

 switch (temp) //如果临时变量的值为wasd中的一个,则赋值给key
 {
 default:
 break;
 case 'w':
 case 'a':
 case 's':
 case 'd':
 //如果temp的方向和key的方向不相反则赋值
 if ((key == 'w' && temp != 's') || (key == 's' && temp != 'w') || \
 (key == 'a' && temp != 'd') || (key == 'd' && temp != 'a'))
 key = temp;
 }
 }


 switch (key) //根据key的值确定蛇的移动方向
 {
 case 'd':
 nexthead.x = snakecoor.front().x + 2; //新的蛇头的头部等于容器内第一个数据(旧蛇头)x坐标+2
 nexthead.y = snakecoor.front().y;
 break;
 case 'a':
 nexthead.x = snakecoor.front().x - 2;
 nexthead.y = snakecoor.front().y;
 break;
 case 'w':
 nexthead.x = snakecoor.front().x;
 nexthead.y = snakecoor.front().y - 1;
 break;
 case 's':
 nexthead.x = snakecoor.front().x;
 nexthead.y = snakecoor.front().y + 1;
 }

 }

 //游戏结束时需要做的事情
 void finmatt(const int score)
 {
 system("cls");
 gotoxy(40, 14);
 cout << "游戏结束";
 gotoxy(40, 16);
 cout << "得分:" << score;
 gotoxy(0, 26);
 exit(0);
 }

 //游戏结束的情况
 void finishgame(const int score)
 {
 //撞墙情况
 if (snakecoor[0].x >= 88 || snakecoor[0].x < 0 || snakecoor[0].y >= 28 || snakecoor[0].y < 0)
 finmatt(score);

 //撞到自己情况
 for (int i = 1; i < snakecoor.size(); i++)
 if (snakecoor[0].x == snakecoor[i].x && snakecoor[0].y == snakecoor[i].y)
 finmatt(score);
 }
public:
 //构造初始化蛇的位置
 Snake()
 {
 Snakecoor temp; //临时结构变量用于创建蛇

 for (int i = 5; i >= 0; i--) //反向创建初始蛇身,初始蛇头朝东
 {
 temp.x = 16 + (i << 1); //偶数
 temp.y = 8;
 snakecoor.push_back(temp);
 }

 }

 //蛇运动主要函数
 void move(Food& food, int& score)
 {
 Snakecoor nexthead; //新蛇头变量

 degdir(nexthead); //判断和改变蛇前进的方向

 snakecoor.insert(snakecoor.begin(), nexthead); //将新的蛇头插入容器头部

 gotoxy(0, 0);
 cout << "得分:" << score; //每次移动都在左上角刷新得分

 finishgame(score); //判断游戏结束函数

 if (snakecoor[0].x == food.getFoodm_x() && snakecoor[0].y == food.getFoodm_y()) //蛇头与食物重合
 {
 gotoxy(snakecoor[0].x, snakecoor[0].y); //吃到食物时因为直接返回此次移动没有输出蛇身,会少输出一次蛇
 cout << "●"; //所以在这里补上蛇移动时需要输出的字符
 gotoxy(snakecoor[1].x, snakecoor[1].y);
 cout << "■";

 score++; //吃到食物得分+1

 food.randfood(); //如果蛇头坐标和食物坐标重合则重新产生一个食物
 return; //直接结束本次移动
 }

 for (int i = 0; i < snakecoor.size(); i++) //遍历容器,判断食物与蛇身是否重合并输出整条蛇
 {
 gotoxy(snakecoor[i].x, snakecoor[i].y);

 if (!i) //头部输出圆形否则输出方块
 cout << "●";
 else
 cout << "■";

 //如果食物刷新到了蛇身上,则重新产生一个食物
 if (snakecoor[i].x == food.getFoodm_x() && snakecoor[i].y == food.getFoodm_y())
 food.randfood();
 }

 gotoxy(snakecoor.back().x, snakecoor.back().y); //在容器尾部的地方输出空格
 cout << " ";
 snakecoor.pop_back(); //删除容器中最后一个数据
 }

};


void HideCursor() //隐藏光标
{
 CONSOLE_CURSOR_INFO cursor_info = { 1, 0 };
 SetConsoleCursorInfo(GetStdHandle(STD_OUTPUT_HANDLE), &cursor_info);
}

void gotoxy(int x, int y) //光标定位
{
 COORD pos = { x,y };
 SetConsoleCursorPosition(GetStdHandle(STD_OUTPUT_HANDLE), pos);
}

//主函数
int main()
{
 system("mode con cols=88 lines=28"); //设置控制台窗口大小
 system("title C++ 贪吃蛇");  //设置标题
 HideCursor();  //光标隐藏

 int score = 0; //得分变量

 Food food; //食物对象
 food.randfood(); //开局随机产生一个食物

 Snake snake; //蛇对象


 while (true)
 {
 snake.move(food, score);//蛇移动

 Sleep(150); //游戏速度
 }

 return 0;
}

不足之处:

因为初学C++,所以程序中肯定还有一些不规范或不合理的地方。

关于C++小游戏的更多精彩内容请点击专题: 《C++经典小游戏》 学习了解

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

更多精彩内容其他人还在看

用标准c++实现string与各种类型之间的转换

这个类在头文件中定义, < sstream>库定义了三种类:istringstream、ostringstream和stringstream,分别用来进行流的输入、输出和输入输出操作。另外,每个类都有一个对应的宽字符集版本
收藏 0 赞 0 分享

C++如何通过ostringstream实现任意类型转string

再使用整型转string的时候感觉有点棘手,因为itoa不是标准C里面的,而且即便是有itoa,其他类型转string不是很方便。后来去网上找了一下,发现有一个好方法
收藏 0 赞 0 分享

C/C++指针小结

要搞清一个指针需要搞清指针的四方面的内容:指针的类型,指针所指向的类型,指针的值或者叫指针所指向的内存区,还有指针本身所占据的内存区
收藏 0 赞 0 分享

C++ 类的静态成员深入解析

在C++中类的静态成员变量和静态成员函数是个容易出错的地方,本文先通过几个例子来总结静态成员变量和成员函数使用规则,再给出一个实例来加深印象
收藏 0 赞 0 分享

C++类的静态成员初始化详细讲解

通常静态数据成员在类声明中声明,在包含类方法的文件中初始化.初始化时使用作用域操作符来指出静态成员所属的类.但如果静态成员是整型或是枚举型const,则可以在类声明中初始化
收藏 0 赞 0 分享

C++类静态成员与类静态成员函数详解

静态成员不可在类体内进行赋值,因为它是被所有该类的对象所共享的。你在一个对象里给它赋值,其他对象里的该成员也会发生变化。为了避免混乱,所以不可在类体内进行赋值
收藏 0 赞 0 分享

C++中的friend友元函数详细解析

友元可以是一个函数,该函数被称为友元函数;友元也可以是一个类,该类被称为友元类。友元函数的特点是能够访问类中的私有成员的非成员函数。友元函数从语法上看,它与普通函数一样,即在定义上和调用上与普通函数一样
收藏 0 赞 0 分享

static全局变量与普通的全局变量的区别详细解析

以下是对static全局变量与普通的全局变量的区别进行了详细的分析介绍,需要的朋友可以过来参考下,希望对大家有所帮助
收藏 0 赞 0 分享

C++ explicit关键字的应用方法详细讲解

C++ explicit关键字用来修饰类的构造函数,表明该构造函数是显式的,既然有"显式"那么必然就有"隐式",那么什么是显示而什么又是隐式的呢?下面就让我们一起来看看这方面的知识吧
收藏 0 赞 0 分享

教你5分钟轻松搞定内存字节对齐

随便google一下,人家就可以跟你解释的,一大堆的道理,我们没怎么多时间,讨论为何要对齐.直入主题,怎么判断内存对齐规则,sizeof的结果怎么来的,请牢记以下3条原则
收藏 0 赞 0 分享
查看更多