(七)C++自制植物大战僵尸游戏关卡数据加载代码讲解

植物大战僵尸游戏开发教程专栏地址icon-default.png?t=N7T8http://t.csdnimg.cn/xjvbb 


打开LevelData.h和LevelData.cpp文件。文件位置如下图所示。

 LevelData.h

此头文件中定义了两个类,分别是OpenLevelData、LevelData,其中OpenLevelData用于加载文件数据。LevelData解析数据,将数据保存到数据结构中。

#pragma once
#include "cocos2d.h"
#include "json/writer.h"
#include "json/document.h"
#include "json/stringbuffer.h"using namespace std;
using namespace cocos2d;
using namespace rapidjson;class LevelData;class OpenLevelData
{
public:/***单例*/static OpenLevelData* getInstance();/***打开关卡数据*/bool openLevelsData(const string& worlddata);/**解密关卡数据*/void decrypt(char* cSrc, char* cDest);void decrypt(string& cSrc, char* cDest);/***获取所有关卡数据*/Document* getDocument();/***创建某一个关卡数据*/void createLevelData(const int level, const char* levelName);/***读取某一个关卡*/LevelData* readLevelData(const int level);/***设置关卡数*/void setLevelNumber(const int levelNumber);/***获取关卡数*/int getLevelNumber() const;/**初始化*/void documentInit();private:OpenLevelData():_document(new Document), _levelNumber(-1){}~OpenLevelData() {}private:static OpenLevelData* _instance;Document* _document;map<int, LevelData*>_levelData;int _levelNumber;
};struct MyPoint
{MyPoint():x(0),y(0){}int x, y;
};class LevelData
{
public:bool readLevelData(const char* LevelName);bool getZombiesVisible() const { return _zombiesIsVisible; }bool getZombiesIsSmall() const { return _zombiesIsSmall; }bool getZombiesIsBig() const { return _zombiesIsBig; }bool getIsNoPlants() const { return _isNoPlants; }int getZombiesFrequency() const { return _zombiesFrequency; }int getCoinNumbers() const { return _coinNumbers; }int getAtLeastSunNumbers() const { return _atLeastSunNumbers; }int getFlowerPosition() const { return _flowerPosition; }int getCarNumbers() const { return _carNumbers; }int getUsePlantsNumbers() const { return _usePlantsNumbers; }int getFirstFrequencyTime() const { return _firstFrequencyTime; }float getUserLostPosition() const { return _userLose; }vector<int>& getGameType() { return _gameType; }vector<int>& getZombiesType() { return _zombiesType; }vector<int>& getZombiesNumbers() { return _zombiesNumbers; }vector<int>& getMunchZombiesFrequency() { return _munchZombiesFrequency; }vector<MyPoint>& getNoPlantsPosition() { return _noPlantsPosition; }vector<vector<int> >& getZombiesTypeProbabilityFrequency() { return _zombiesTypeProbabilityFrequency; }CC_CONSTRUCTOR_ACCESS:LevelData();~LevelData();private:void setGameTypes(const char* LevelName);private:bool _isEncryption;                                              /* 是否加密 */bool _zombiesIsVisible;                                          /* 僵尸是否隐身 */bool _zombiesIsSmall;                                            /* 是否是小僵尸 */bool _zombiesIsBig;                                              /* 是否是巨人僵尸 */bool _isNoPlants;                                                /* 是否不可种植 */int _zombiesFrequency;                                           /* 僵尸总波数 */int _coinNumbers;                                                /* 金币数 */int _atLeastSunNumbers;                                          /* 至少产生的阳光数 */int _flowerPosition;                                             /* 花坛位置 */int _carNumbers;                                                 /* 小车数量 */int _usePlantsNumbers;                                           /* 使用植物数量 */int _firstFrequencyTime;                                         /* 第一波僵尸出现时间 */float _userLose;                                                 /* 玩家失败 */vector<int>_gameType;                                            /* 游戏类型 */vector<int>_zombiesType;                                         /* 僵尸类型 */vector<int>_zombiesNumbers;                                      /* 僵尸数 */vector<int>_munchZombiesFrequency;                               /* 多僵尸波数 */vector<vector<int> >_zombiesTypeProbabilityFrequency;            /* 每一波每种僵尸出现的概率 */vector<MyPoint>_noPlantsPosition;                                /* 不可以种植的地方 */Document* _document;
};

LevelData.cpp

getInstance()函数

OpenLevelData使用了单例模式,这样保证了只会创建唯一的一个实例。该实例会使用map数据结构保存所有关卡数据。

map<int, LevelData*>_levelData;
OpenLevelData* OpenLevelData::getInstance()
{if (_instance == nullptr){_instance = new (std::nothrow)OpenLevelData;}return _instance;
}

openLevelsData()函数

函数有一个参数,表示要加载的文件名称。使用Cocos2d-xFileUtils类加载磁盘文件中的数据,函数返回字符串类型数据。

由于该文件存在加密,所以首先对字符串数据进行解密,然后使用RapidJson库来解析json字符串数据。如果解析失败返回false,解析成功返回true

bool OpenLevelData::openLevelsData(const string& worlddata)
{char* passWords;string str = FileUtils::getInstance()->getStringFromFile(worlddata);passWords = (char*)malloc(sizeof(char) * str.size());/* 解密 */decrypt(str, passWords);documentInit();_document->Parse<rapidjson::kParseDefaultFlags>(passWords);free(passWords);if (_document->HasParseError())return false;return true;
}

decrypt()函数

函数有两个参数,第一个参数表示要解密的字符串,第二个参数表示解密后的字符串。通过逐个遍历字符进行字符串解密,其算法如下代码所示。

void OpenLevelData::decrypt(string& cSrc, char* cDest)
{int   i, h, l, m, n, j = 0;for (i = 0; i < static_cast<int>(cSrc.size()); i = i + 2){h = (cSrc[i] - 'x');l = (cSrc[i + 1] - 'z');m = (h << 4);n = (l & 0xf);cDest[j] = m + n;j++;}cDest[j] = '\0';
}

createLevelData()函数

函数有两个参数,第一参数表示关卡编号,用作map中的key值,当需要获取某一关卡数据时,只需要根据key值就可以获取相关数据。第二参数是关卡名称,根据关卡名称获取文件中的关卡数据。

void OpenLevelData::createLevelData(const int level, const char* levelName)
{/* map中如果没有关卡数据 */if (!_levelData.count(level)){LevelData* levelData = new LevelData;levelData->readLevelData(levelName);_levelData.insert(pair<int, LevelData*>(level, levelData));}
}

这个函数功能是解析某一关卡数据,并将其保存的map数据结构中。首先判断map数据结构中是否已经存在该关卡数据,如果不存在,则使用LevelData类中的readLevelData()函数解析文件中该关卡的数据然后将其保存到map数据结构中供后续使用。


readLevelData()函数

函数有一个参数,表示关卡名称,函数根据关卡名称来解析json文件。

bool LevelData::readLevelData(const char* LevelName)
{_document = OpenLevelData::getInstance()->getDocument();if (_document->HasMember(LevelName)){_isEncryption = (*_document)[LevelName]["IsEncryption"].GetBool();_coinNumbers = (*_document)[LevelName]["CoinNumbers"].GetInt();_zombiesFrequency = (*_document)[LevelName]["Frequency"].GetInt();_firstFrequencyTime = (*_document)[LevelName]["FirstFrequencyTime"].GetInt();_userLose = (*_document)[LevelName]["UserLose"].GetFloat();for (unsigned int i = 0; i < (*_document)[LevelName]["GameType"].Size(); i++){_gameType.push_back((*_document)[LevelName]["GameType"][i].GetInt());}for (unsigned int i = 0; i < (*_document)[LevelName]["ZombiesType"].Size(); i++){_zombiesType.push_back((*_document)[LevelName]["ZombiesType"][i].GetInt());}for (unsigned int i = 0; i < (*_document)[LevelName]["MunchZombiesFrequency"].Size(); i++){_munchZombiesFrequency.push_back((*_document)[LevelName]["MunchZombiesFrequency"][i].GetInt());}for (unsigned int i = 0; i < (*_document)[LevelName]["ZombiesNumbers"].Size(); i++){_zombiesNumbers.push_back((*_document)[LevelName]["ZombiesNumbers"][i].GetInt());}vector<int> v;for (unsigned int i = 0; i < (*_document)[LevelName]["ZombiesTypeProbability"].Size(); i++){v.clear();for (unsigned int j = 0; j < (*_document)[LevelName]["ZombiesTypeProbability"][i].Size(); j++){v.push_back((*_document)[LevelName]["ZombiesTypeProbability"][i][j].GetInt());}_zombiesTypeProbabilityFrequency.push_back(v);}setGameTypes(LevelName);return true;}return false;
}

函数首先判断json文件中是否存在以形参变量命名的关卡名称,如果有则进行数据解析,最后返回true表示解析成功,否则返回false表示解析失败。


setGameTypes()函数

函数有一个参数,表示关卡名称,函数根据关卡名称来解析json文件。

void LevelData::setGameTypes(const char* LevelName)
{for (unsigned int i = 0; i < _gameType.size(); i++){switch (static_cast<GameTypes>(_gameType.at(i))){case GameTypes::AtLeastSunNumbers:_atLeastSunNumbers = (*_document)[LevelName]["AtLeastSunNumbers"].GetInt();break;case GameTypes::FlowerPosition:_flowerPosition = 570 + 122 * (*_document)[LevelName]["FlowerPosition"].GetInt();break;case GameTypes::CarNumbers:_carNumbers = (*_document)[LevelName]["CarNumbers"].GetInt();break;case GameTypes::UserPlantsNumbers:_usePlantsNumbers = (*_document)[LevelName]["UserPlantsNumbers"].GetInt();break;case GameTypes::ZombiesInvisible:_zombiesIsVisible = true;break;case GameTypes::SmallZombies:_zombiesIsSmall = true;break;case GameTypes::BigZombies:_zombiesIsBig = true;break;case GameTypes::NoPlants:{_isNoPlants = true;MyPoint MyPoint;for (unsigned int i = 0; i < (*_document)[LevelName]["NoPlants"].Size(); i++){MyPoint.x = (*_document)[LevelName]["NoPlants"][i][0].GetInt();MyPoint.y = (*_document)[LevelName]["NoPlants"][i][1].GetInt();_noPlantsPosition.push_back(MyPoint);}}break;}}
}

函数根据不同的游戏类型解析不同的数据,使用switch case语句来判断不同的游戏类型。 


其他函数

其他函数就不一一介绍了,可以自行查看阅读代码。

后续

后续将讲解游戏多语言切换功能的实现。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

vue中预览docx、xlsx、pptx、pdf

前言&#xff1a;其实本来是要做全类型文件预览的&#xff0c;但是一直找不到合适的doc,xlx,ppt预览插件。要是有可以使用的&#xff0c;可以评论推荐给我 我使用的node版本&#xff1a;v18.19.1 参考官网&#xff1a;preview 文件预览 | ran 引入方式&#xff1a; //安装组…

学习笔记之——3DGS-SLAM系列代码解读

最近对一系列基于3D Gaussian Splatting&#xff08;3DGS&#xff09;SLAM的工作的源码进行了测试与解读。为此写下本博客mark一下所有的源码解读以及对应的代码配置与测试记录~ 其中工作1~5的原理解读见博客&#xff1a; 学习笔记之——3D Gaussian Splatting及其在SLAM与自动…

Qt 中默认代码

目录 主函数 widget的声明 widget的定义 form file .pro 文件 主函数 #include "widget.h" ​ #include <QApplication> ​ int main(int argc, char *argv[]) {QApplication a(argc, argv);Widget w;w.show();return a.exec(); } 上面就是 Qt 刚创建的一…

非关系型数据库(缓存数据库)redis的集群

目录 一.群集模式——Cluster 1.原理 2.作用 3.特点 4.工作机制 哈希槽 哈希槽的分配 哈希槽可按照集群主机数平均分配&#xff08;默认分配&#xff09; 根据主机的性能以及功能自定义分配 redis集群的分片 分片 如何找到给定key的分片 优势 二. 搭建Redis群集…

JVM虚拟机(五)强引用、软引用、弱引用、虚引用

目录 一、强引用二、软引用三、弱引用四、虚引用五、总结 引文&#xff1a; 在 Java 中一共存在 4 种引用&#xff1a;强、软、弱、虚。它们主要指的是&#xff0c;在进行垃圾回收的时候&#xff0c;对于不同的引用垃圾回收的情况是不一样的。下面我们就一起来看一下这 4 种引用…

基于R语言实现的负二项回归模型【理解与实现】-理解负二项回归模型和泊松回归模型之间的区别

前言 我们可以在R语言中使用MASS包中的glm.nb函数来拟合负二项模型&#xff0c;以及使用glm函数来拟合泊松模型。以下是一个详细的过程&#xff0c;包括模拟数据的生成、模型的拟合、结果的比较和解释。 需要的包 if (!require("MASS")) install.packages("M…

Linux网络基础 (二) ——(IP、MAC、端口号、TCPUDP协议、网络字节序)

文章目录 IP 地址基本概念源IP地址 & 目的IP地址 MAC 地址基本概念源MAC地址 & 目的MAC地址 端口号基本概念源端口号 & 目的端口号 TCP & UDP 协议基本概念TCP 与 UDP 的抉择 网络字节序大端、小端字节序 &#x1f396; 博主的CSDN主页&#xff1a;Ryan.Alask…

YOLOv8模型剪枝实战:DepGraph(依赖图)方法

课程链接&#xff1a;YOLOv8模型剪枝实战&#xff1a;DepGraph(依赖图)方法_在线视频教程-CSDN程序员研修院 YOLOv8是一个当前非常流行的目标检测器&#xff0c;本课程使用DepGraph&#xff08;依赖图&#xff09;剪枝方法对YOLOv8进行网络剪枝&#xff0c;使其更加轻量和实用…

Qt快速入门(MV架构之TableView + QStandardItemModel + 自定义代理小案例)

Qt快速入门&#xff08;MV架构之TableView QStandardItemModel 自定义代理小案例&#xff09; 关于MV架构的简单介绍 在Qt框架中&#xff0c;代理&#xff08;Delegate&#xff09;、模型&#xff08;Model&#xff09;和视图&#xff08;View&#xff09;之间的关系构成了…

51单片机实验03-单片机定时/计数器实验

目录 一、实验目的 二、实验说明 1、51单片机有两个16位内部计数器/定时器&#xff08;C/T&#xff0c; Counter/Timer&#xff09;。 2、模式寄存器TMOD 1) M1M0工作模式控制位&#xff1b; 2) C/T定时器或计数器选择位&#xff1a; 3&#xff09;GATE定时器/计数器运行…

Python零基础从小白打怪升级中~~~~~~~模块+异常+Pycharm的debug调试

第十节&#xff1a;模块异常Debug 一、Python模块 Python 模块(Module)&#xff0c;是一个 Python 文件&#xff0c;以 .py 结尾&#xff0c;模块能定义函数&#xff0c;类和变量&#xff0c;模块里也能包含可执行的代码。 1、导入模块 导入模块的5中方式 import 模块名fr…

C++ 之 【类与对象】从入门到精通一条龙服务 最终篇(static成员、友元、匿名对象。。。)

&#x1f4b4;到用时方恨早&#xff0c;白首方悔挣的少 车到山前没有路&#xff0c;悬崖勒马勒不住 一、再谈构造函数 1.构造函数体赋值 2.初始化列表 3.explicit关键字 二、Static成员 1.概念 2.特性 三、友元 1.友元函数 2.友元类 四、内部类 五、匿名对象 六、…

结合 tensorflow.js 、opencv.js 与 Ant Design 创建美观且高性能的人脸动捕组件并发布到InsCode

系列文章目录 如何在前端项目中使用opencv.js | opencv.js入门如何使用tensorflow.js实现面部特征点检测tensorflow.js 如何从 public 路径加载人脸特征点检测模型tensorflow.js 如何使用opencv.js通过面部特征点估算脸部姿态并绘制示意图tensorflow.js 使用 opencv.js 将人脸…

模型预测控制MPC(1)—— 基础概念

文章目录 1. 历史沿革1.1 控制论1.2 最优控制与强化学习 2. 模型预测控制 1. 历史沿革 我本科做机器人的时候接触过一点控制&#xff0c;主要做了大量 PID 在嵌入式控制系统的应用&#xff1b;硕士期间研究方向是强化学习。在我调研 MPC 的过程中&#xff0c;发现它同时出现在…

云笔记小程序的实现

1.前言 云笔记, 是基于HotApp小程序统计云后台提供的api接口开发的一个微信小程序。 2.功能 离线保存笔记 云端数据同步, 更换了设备也可以找到以前的笔记 接入了好推二维码提供的数据统计工具, 可以到平台上查看用户分析、留存分析、事件分析。 3.界面效果 ***HotApp云笔…

Redis Pipelining 底层原理分析及实践

作者&#xff1a;vivo 互联网服务器团队-Wang Fei Redis是一种基于客户端-服务端模型以及请求/响应的TCP服务。在遇到批处理命令执行时&#xff0c;Redis提供了Pipelining(管道)来提升批处理性能。本文结合实践分析了Spring Boot框架下Redis的Lettuce客户端和Redisson客户端对…

行云堡垒国密算法应用与信创支持

一、 国密算法和信创的介绍 1.1 什么是国密算法 国密算法是国家密码管理局制定颁布的一系列的密码标准&#xff0c;即已经被国家密码局认定的国产密码算法&#xff0c;又称商用密码&#xff08;是指能够实现商用密码算法的加密&#xff0c;解密和认证等功能的技术&#xff09;…

excel里如何的科学计数法的数字转换成数值?

比如下图&#xff0c;要想把它们转换成3250跟1780&#xff0c;有什么快捷的办法吗&#xff1f; 科学计数法在excel里的格式&#xff0c;与我们常规在数学上写的有差异。这个转换可以这样做&#xff1a; 1.转换后的效果&#xff1a; 2.问题分析 题目中所附截图&#xff0c;单元…

Day96:云上攻防-云原生篇Docker安全系统内核版本漏洞CDK自动利用容器逃逸

目录 云原生-Docker安全-容器逃逸&系统内核漏洞 云原生-Docker安全-容器逃逸&docker版本漏洞 CVE-2019-5736 runC容器逃逸(需要管理员配合触发) CVE-2020-15257 containerd逃逸(启动容器时有前提参数) 云原生-Docker安全-容器逃逸&CDK自动化 知识点&#xff1…

Redis入门到通关之Set命令

文章目录 ⛄ 概述⛄ Set类型的常见命令⛄RedisTemplate API❄️❄️ 添加Set缓存(值可以是一个&#xff0c;也可是多个)❄️❄️设置过期时间(单独设置)❄️❄️根据key获取Set中的所有值❄️❄️根据value从一个set中查询,是否存在❄️❄️获取Set缓存的长度❄️❄️移除指定的…