(七)C++自制植物大战僵尸游戏关卡数据加载代码讲解
植物大战僵尸游戏开发教程专栏地址http://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-x中FileUtils类加载磁盘文件中的数据,函数返回字符串类型数据。
由于该文件存在加密,所以首先对字符串数据进行解密,然后使用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语句来判断不同的游戏类型。
其他函数
其他函数就不一一介绍了,可以自行查看阅读代码。
后续
后续将讲解游戏多语言切换功能的实现。
相关文章:

(七)C++自制植物大战僵尸游戏关卡数据加载代码讲解
植物大战僵尸游戏开发教程专栏地址http://t.csdnimg.cn/xjvbb 打开LevelData.h和LevelData.cpp文件。文件位置如下图所示。 LevelData.h 此头文件中定义了两个类,分别是OpenLevelData、LevelData,其中OpenLevelData用于加载文件数据。LevelData解析数据…...

wpf下RTSP|RTMP播放器两种渲染模式实现
技术背景 在这篇blog之前,我提到了wpf下播放RTMP和RTSP渲染的两种方式,一种是通过控件模式,另外一种是直接原生RTSP、RTMP播放模块,回调rgb,然后在wpf下渲染,本文就两种方式做个说明。 技术实现 以大牛直…...

Element-UI 自定义-下拉框选择年份
1.实现效果 场景表达: 默认展示当年的年份,默认展示前7年的年份 2.实现思路 创建一个新的Vue组件。 使用<select>元素和v-for指令来渲染年份下拉列表。 使用v-model来绑定选中的年份值。 3.实现代码展示 <template><div><el-…...
二叉树的链式存储
二叉树是一种非常重要的数据结构,它能够高效地进行数据的插入、删除和查找操作。二叉树的每个节点最多有两个子节点,分别是左子节点和右子节点。二叉树可以采用多种不同的存储方式来实现,其中链式存储是最为直观和常用的一种方法。本文将深入…...

[计算机效率] 鼠标手势工具:WGestures(解放键盘的超级效率工具)
3.22 鼠标手势工具:WGestures 通过设置各种鼠标手势和操作进行绑定。当用户通过鼠标绘制出特定的鼠标手势后就会触发已经设置好的操作。有点像浏览器中的鼠标手势,通过鼠标手势操纵浏览器做一些特定的动作。这是一款强大的鼠标手势工具,可以…...
Linux useradd命令教程:如何创建新的用户账户(附实例详解和注意事项)
Linux useradd命令介绍 useradd是Linux中用于添加用户账户的命令。它可以用于创建新的用户,并可以配合不同的选项来指定用户的主目录、UID、GID、组等信息。 Linux useradd命令适用的Linux版本 useradd命令在大多数Linux发行版中都可以使用,包括但不限…...

基于ollama搭建本地chatGPT
ollama帮助我们可以快速在本地运行一个大模型,再整合一个可视化页面就能构建一个chatGPT,可视化页面我选择了chat-ollama(因为它还能支持知识库,可玩性更高),如果只是为了聊天更推荐chatbox 部署步骤 下载…...

C++11 数据结构3 线性表的循环链式存储,实现,测试
上一节课,我们学了线性表 单向存储结构(也就是单链表),这个是企业常用的技术,且是后面各种的基本,一定要牢牢掌握,如果没有掌握,下面的课程会云里雾里。 一 ,循环链表 1…...

初识DOM
目录 前言: 1.初识DOM: 1.1DOM树: 1.2节点(Node): 1.2.1元素节点: 1.2.2属性节点: 1.2.3文本节点: 1.3Document对象: 2.操作网页元素: 2.1找出元素: 2.1.1document.getElementById(id)࿱…...

计算机视觉实验五——图像分割
计算机视觉实验五——图像分割 一、实验目标二、实验内容1.了解图割操作,实现用户交互式分割,通过在一幅图像上为前景和背景提供一些标记或利用边界框选择一个包含前景的区域,实现分割①图片准备②代码③运行结果④代码说明 2.采用聚类法实现…...

移动Web学习06-移动端适配Less预处理器项目案例
项目目标:实现在不同宽度设备中等比缩放的网页效果 Less代码 import ./base; import ./normalize;// 变量: 存储37.5 rootSize: 37.5rem; *{margin: 0;padding: 0; } body {background-color: #F0F0F0; }// 主体内容 .main {// padding-bottom: (50 / 37.5rem);pa…...

LangChain-25 ReAct 让大模型自己思考和决策下一步 AutoGPT实现途径、AGI重要里程碑
背景介绍 大模型ReAct(Reasoning and Acting)是一种新兴的技术框架,旨在通过逻辑推理和行动序列的构建,使大型语言模型(LLM)能够达成特定的目标。这一框架的核心思想是赋予机器模型类似人类的推理和行动能…...
24/04/15总结
多线程: 线程 线程是操作系统能够进行运算调度的最小单位。它被包含在进程之中,是进程中的实际运作单位 并发:在同一时刻,有多个指令在单个cpu上交替执行 并行:在同一时刻,有多个指令在多个cpu上同时执行 多线程的实现方式 1.继承…...
vue3、vue2中nextTick源码解析
nexttick是啥 nextTick是Vue提供的一个全局API,由于Vue的异步更新策略导致我们对数据的修改不会更新,如果此时想要获取更新后的Dom,就需要使用这个方法. vue的异步更新策略意思是如果数据变化,vue不会立刻更新dom,而是开启一个队列,把组件更…...

【氮化镓】GaN HEMTs结温和热阻测试方法
文章《Temperature rise detection in GaN high-electron-mobility transistors via gate-drain Schottky junction forward-conduction voltages》,由Xiujuan Huang, Chunsheng Guo, Qian Wen, Shiwei Feng, 和 Yamin Zhang撰写,发表在《Microelectroni…...

c++11 标准模板(STL)本地化库 - 平面类别(std::codecvt) - 在字符编码间转换,包括 UTF-8、UTF-16、UTF-32 (四)
本地化库 本地环境设施包含字符分类和字符串校对、数值、货币及日期/时间格式化和分析,以及消息取得的国际化支持。本地环境设置控制流 I/O 、正则表达式库和 C 标准库的其他组件的行为。 平面类别 在字符编码间转换,包括 UTF-8、UTF-16、UTF-32 std::…...

【状态压缩 容斥原理 组合数学】100267. 单面值组合的第 K 小金额
本文涉及知识点 状态压缩 容斥原理 组合数学 二分查找算法合集 LeetCode100267. 单面值组合的第 K 小金额 给你一个整数数组 coins 表示不同面额的硬币,另给你一个整数 k 。 你有无限量的每种面额的硬币。但是,你 不能 组合使用不同面额的硬币。 返回…...

.net框架和c#程序设计第三次测试
目录 一、测试要求 二、实现效果 三、实现代码 一、测试要求 二、实现效果 数据库中的内容: 使用数据库中的账号登录: 若不是数据库中的内容: 三、实现代码 login.aspx文件: <% Page Language"C#" AutoEventW…...

架构师系列-搜索引擎ElasticSearch(五)- 索引设计
索引创建后,要非常谨慎,创建不好后面会出现各种问题。 索引设计的重要性 索引创建后,索引分片只能通过_split和_shrink 接口对其进行成倍的增加和缩减。 ES的数据是通过_routing分配到各个分片上的,所以本质上不推荐区改变索引的…...
kafka ----修改log4j、jmx、jvm参数等
1、修改log4j 日志路径 在kafka-run-class.sh文件中修改如下配置,将 LOG_DIR变量指定为自己想要存储的路径 # Log directory to use if [ "x$LOG_DIR" "x" ]; thenLOG_DIR"$base_dir/logs" fi2、修改jmx参数 在kafka-run-class.s…...

网络编程(Modbus进阶)
思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…...

第19节 Node.js Express 框架
Express 是一个为Node.js设计的web开发框架,它基于nodejs平台。 Express 简介 Express是一个简洁而灵活的node.js Web应用框架, 提供了一系列强大特性帮助你创建各种Web应用,和丰富的HTTP工具。 使用Express可以快速地搭建一个完整功能的网站。 Expre…...

以下是对华为 HarmonyOS NETX 5属性动画(ArkTS)文档的结构化整理,通过层级标题、表格和代码块提升可读性:
一、属性动画概述NETX 作用:实现组件通用属性的渐变过渡效果,提升用户体验。支持属性:width、height、backgroundColor、opacity、scale、rotate、translate等。注意事项: 布局类属性(如宽高)变化时&#…...
深入浅出:JavaScript 中的 `window.crypto.getRandomValues()` 方法
深入浅出:JavaScript 中的 window.crypto.getRandomValues() 方法 在现代 Web 开发中,随机数的生成看似简单,却隐藏着许多玄机。无论是生成密码、加密密钥,还是创建安全令牌,随机数的质量直接关系到系统的安全性。Jav…...

基于Flask实现的医疗保险欺诈识别监测模型
基于Flask实现的医疗保险欺诈识别监测模型 项目截图 项目简介 社会医疗保险是国家通过立法形式强制实施,由雇主和个人按一定比例缴纳保险费,建立社会医疗保险基金,支付雇员医疗费用的一种医疗保险制度, 它是促进社会文明和进步的…...

DAY 47
三、通道注意力 3.1 通道注意力的定义 # 新增:通道注意力模块(SE模块) class ChannelAttention(nn.Module):"""通道注意力模块(Squeeze-and-Excitation)"""def __init__(self, in_channels, reduction_rat…...
Neo4j 集群管理:原理、技术与最佳实践深度解析
Neo4j 的集群技术是其企业级高可用性、可扩展性和容错能力的核心。通过深入分析官方文档,本文将系统阐述其集群管理的核心原理、关键技术、实用技巧和行业最佳实践。 Neo4j 的 Causal Clustering 架构提供了一个强大而灵活的基石,用于构建高可用、可扩展且一致的图数据库服务…...
大模型多显卡多服务器并行计算方法与实践指南
一、分布式训练概述 大规模语言模型的训练通常需要分布式计算技术,以解决单机资源不足的问题。分布式训练主要分为两种模式: 数据并行:将数据分片到不同设备,每个设备拥有完整的模型副本 模型并行:将模型分割到不同设备,每个设备处理部分模型计算 现代大模型训练通常结合…...
IP如何挑?2025年海外专线IP如何购买?
你花了时间和预算买了IP,结果IP质量不佳,项目效率低下不说,还可能带来莫名的网络问题,是不是太闹心了?尤其是在面对海外专线IP时,到底怎么才能买到适合自己的呢?所以,挑IP绝对是个技…...

[免费]微信小程序问卷调查系统(SpringBoot后端+Vue管理端)【论文+源码+SQL脚本】
大家好,我是java1234_小锋老师,看到一个不错的微信小程序问卷调查系统(SpringBoot后端Vue管理端)【论文源码SQL脚本】,分享下哈。 项目视频演示 【免费】微信小程序问卷调查系统(SpringBoot后端Vue管理端) Java毕业设计_哔哩哔哩_bilibili 项…...