【Unity】RPG2D龙城纷争(二)关卡、地块
更新日期:2024年6月12日。
项目源码:后续章节发布
索引
- 简介
- 地块(Block)
- 一、定义地块类
- 二、地块类型
- 三、地块渲染
- 四、地块索引
- 关卡(Level)
- 一、定义关卡类
- 二、关卡基础属性
- 三、地块集合
- 四、关卡初始化
- 五、关卡销毁
- 六、回合制逻辑
简介
本章我们将从零开始
实现关卡
,不过,我们的目的是实现关卡与地块
的基本逻辑,更复杂的功能我们打算在开发关卡编辑器
时再做涉及,正所谓一步一个坎,就没有迈不过去的高山。
地块(Block)
地块作为关卡的组成元素,他是一个正方形的格子(当然也可以是菱形,针对斜视角游戏,不过这不在框架的支持范畴内,但做出一定的修改即可实现),每一个地块都有属于他自己的一些属性。
一、定义地块类
首先,我们定义地块类Block
:
/// <summary>/// 地块/// </summary>[DisallowMultipleComponent]public class Block : HTBehaviour{}
Block
继承至HTBehaviour
,使得它可以挂载到游戏物体上,然后作为一个地块对象(为不需要一个物体上多次挂载的组件定义DisallowMultipleComponent是一个好习惯,这将提高容错率,如果你的代码会给别人使用的话)。
很多同学热衷于在类继承链上规避
MonoBehaviour
,其实完全没有这个必要,只要你不滥用MonoBehaviour
的生命周期,它将提供给你几大优势:
1.属性可编辑(Inspector
面板);
2.属性可调试(Inspector
面板);
3.对象可追踪(Scene
里面有则有,无则无);
4.对象可销毁(当确定不再使用时,Destroy
它,而不用置null后交给GC)。
二、地块类型
经过深思熟虑,我们将地块类型划分为如下几种:
- 地面
- 山体
- 森林
- 湖泊
- 雪地
- 障碍
编写代码:
/// <summary>/// 地块/// </summary>[DisallowMultipleComponent]public class Block : HTBehaviour{/// <summary>/// 类型/// </summary>[Label("类型")] public BlockType Type;}/// <summary>
/// 地块类型
/// </summary>
public enum BlockType
{/// <summary>/// 地面/// </summary>[Remark("地面")]Ground = 0,/// <summary>/// 山体/// </summary>[Remark("山体")]Moutain = 1,/// <summary>/// 森林/// </summary>[Remark("森林")]Forest = 2,/// <summary>/// 湖泊/// </summary>[Remark("湖泊")]Water = 3,/// <summary>/// 雪地/// </summary>[Remark("雪地")]Snow = 4,/// <summary>/// 障碍/// </summary>[Remark("障碍")]Obstacle = 5
}
然后,我们设计每种类型的地块拥有的交互权限如下:
类型 | 角色可行走 | 角色可攻击(站在上面的敌人或穿过它攻击其他敌人) |
---|---|---|
地面 | 是 | 是 |
山体 | 受限(拥有飞檐走壁可行走,行走速度减1/2 ) | 是 |
森林 | 受限(行走速度减1/2 ) | 是 |
湖泊 | 受限(拥有踏水神行可行走,行走速度减1/2 ) | 是 |
雪地 | 受限(行走速度减2/3 ) | 是 |
障碍 | 否 | 受限(拥有隔山打牛可穿透障碍进行攻击 ) |
需注意的是,任意地块,当上面站有敌人时,将不可行走,不可跨越,当站有队友时,不可行走,可跨越。
也即是说,我们可以使用角色摆出特定的阵型,以拦住敌方的行走路线,或达到包围的效果。
三、地块渲染
地块的渲染我们决定选择SpriteRenderer
,且一个地块仅渲染一张图片,那么,在Block
类中需要持有地块渲染器的引用,我们添加代码:
/// <summary>/// 地块/// </summary>[DisallowMultipleComponent]public class Block : HTBehaviour{/// <summary>/// 目标渲染器/// </summary>[Label("目标渲染器")] public SpriteRenderer Target;/// <summary>/// 类型/// </summary>[Label("类型")] public BlockType Type;}
四、地块索引
然后,我们想一想地块还需要些什么属性,哦对了,我想我们在后续一定会需要检索地块(也即是根据索引找到一个地块),而我们的关卡采用二维平铺布局,所有地块存储的数据结构应当是一个二维数组
最合适,所以地块的索引我们定义为二维的下标Vector2Int
:
/// <summary>/// 地块/// </summary>[DisallowMultipleComponent]public class Block : HTBehaviour{/// <summary>/// 目标渲染器/// </summary>[Label("目标渲染器")] public SpriteRenderer Target;/// <summary>/// 类型/// </summary>[Label("类型")] public BlockType Type;/// <summary>/// 位置/// </summary>[Label("位置")] public Vector2Int Pos;}
地块类的属性暂时就想到这么多,不必追求一次就考虑全面,后续根据情况补充即可,接下来我们定义关卡类。
关卡(Level)
关卡用于容纳并绘制一系列地块
,以及后期容纳角色
等其他的一系列东西。
一、定义关卡类
我们定义关卡类Level
:
/// <summary>/// 关卡/// </summary>[DisallowMultipleComponent]public class Level : SingletonBehaviourBase<Level>{}
Level
继承至单例行为基类SingletonBehaviourBase
,使得它可以挂载到游戏物体上,并作为单例始终全局唯一(同一时刻,场景中的关卡肯定只能有一个)。
二、关卡基础属性
我们先为关卡设计一些基础的属性:
/// <summary>/// 关卡索引/// </summary>[Label("关卡索引")] public int Index;/// <summary>/// 关卡名称/// </summary>[Label("关卡名称")] public string Name;/// <summary>/// 关卡背景音乐/// </summary>[Label("关卡背景音乐")] public AudioClip BGAudio;/// <summary>/// 地图/// </summary>[Label("地图")] public AStarGrid Map;/// <summary>/// 地图尺寸/// </summary> [Label("地图尺寸")] public Vector2Int MapSize;/// <summary>/// 角色根节点/// </summary>[Label("角色根节点")] public Transform RolesRoot;/// <summary>/// 特效根节点/// </summary>[Label("特效根节点")] public Transform EffectsRoot;
属性名称 | 属性详解 |
---|---|
关卡索引 | 关卡唯一标识符,也用作保存、加载关卡时的索引 |
地图 | 所有地块对象的根节点,此处类型为AStarGrid(A*寻路网格) ,兼并寻路功能 |
地图尺寸 | 地图的尺寸,用于描述地图的宽、高 |
角色根节点 | 所有角色对象的根节点 |
特效根节点 | 所有特效对象的根节点 |
将地块、角色、特效
都归于单一的根节点,即方便管理所有对象,又方便统一划层,也即是规定渲染器的遮挡层,这三者的遮挡层关系应当是:特效 > 角色 > 地块
,具体如何实现遮挡,我们先不考虑这么多。
完事后我们的Level在层级面板应当是这样的(三个子对象也即是地块、角色、特效
的根节点):
三、地块集合
定义一个二维数组存储所有地块:
/// <summary>/// 所有的地块/// </summary>public Block[,] Blocks { get; private set; }
此处注意,将其定义为property
的原因是,使其规避序列化
功能(因为不需要序列化,关卡在初始化时搜寻所有地块即可),且提升访问安全性。
四、关卡初始化
然后,定义一个初始化方法
,用于在关卡加载到场景中后,执行他自身的所有初始化操作,此处我们避开MonoBehaviour
的生命周期方法Awake
、Start
,因为在此处他们是不受控的,这也是我上面所提到的不滥用生命周期的另一个意思。
/// <summary>/// 初始化/// </summary>public virtual void Initialize(){//Map为所有地块根节点,即可从Map搜寻所有地块Blocks = new Block[MapSize.x, MapSize.y];Block[] blocks = Map.GetComponentsInChildren<Block>(true);for (int i = 0; i < blocks.Length; i++){//地块的Pos下标,即代表了在二维数组中的索引,我们后续会开发关卡编辑器,Pos的赋值交由编辑器来完成,所以这里只管取Pos值Block block = blocks[i];Blocks[block.Pos.x, block.Pos.y] = block;}}
Initialize
方法作为主动调用方法,在我们自行加载关卡完成后主动调用即可。
五、关卡销毁
同理,应当定义一个销毁方法
,用于在关卡销毁时执行一些操作,虽然我们现在还没有需要做的(地块、角色、特效等都属于关卡物体子节点,会跟着一起销毁,无需额外操作),但事先将其规划好总没错。
/// <summary>/// 销毁/// </summary>public virtual void Dispose(){}
六、回合制逻辑
为了实现回合制逻辑,我们先定义如下几个属性:
/// <summary>/// 当前的回合/// </summary>[PropertyDisplay("当前的回合")]public int Round { get; private set; } = 1;/// <summary>/// 当前回合的行动阵营/// </summary>[PropertyDisplay("当前回合的行动阵营")]public RoleCamp RoundCamp { get; private set; } = RoleCamp.Player;/// <summary>/// 关卡状态/// </summary>[PropertyDisplay("关卡状态")]public LevelState State { get; private set; } = LevelState.InProgress;/// <summary>/// 角色阵营/// </summary>public enum RoleCamp{/// <summary>/// 玩家/// </summary>Player = 0,/// <summary>/// 敌人/// </summary>Enemy = 1}/// <summary>/// 关卡状态/// </summary>public enum LevelState{/// <summary>/// 进行中/// </summary>InProgress,/// <summary>/// 已通关/// </summary>Passed,/// <summary>/// 已失败/// </summary>Failed}
此处应该很好理解了,我们按字面意思来就行了,根据最初的设计,每一个回合:玩家先行动,然后是敌人行动,敌人行动完毕后此回合结束,进入下一回合(循环往复)。
此时,我们发现,完整的回合制逻辑在未编写角色(Role)类前,并不太好写出来,所以我们先放下,将复杂的事情留到后面一步步拆解。
接下来我们准备引入角色(Role)类,不过,看了一眼窗外,今日天色已晚,不宜working…
那么,择日再战吧。
相关文章:
【Unity】RPG2D龙城纷争(二)关卡、地块
更新日期:2024年6月12日。 项目源码:后续章节发布 索引 简介地块(Block)一、定义地块类二、地块类型三、地块渲染四、地块索引 关卡(Level)一、定义关卡类二、关卡基础属性三、地块集合四、关卡初始化五、关…...
mediamtx流媒体服务器测试
MediaMTX简介 在web页面中直接播放rtsp视频流,重点推荐:mediamtx,不仅仅是rtsp-CSDN博客 mediamtx github MediaMTX(以前的rtsp-simple-server)是一个现成的和零依赖的实时媒体服务器和媒体代理,允许发布,读取&…...
C# 循环
C# 循环 在编程中,循环是一种控制结构,它允许我们重复执行一段代码多次。C# 提供了几种循环机制,以适应不同的编程需求。本文将详细介绍 C# 中常用的几种循环类型,包括 for 循环、while 循环、do-while 循环和 foreach 循环&…...
PHP杂货铺家庭在线记账理财管理系统源码
家庭在线记帐理财系统,让你对自己的开支了如指掌,图形化界面操作更简单,非常适合家庭理财、记账,系统界面简洁优美,操作直观简单,非常容易上手。 安装说明: 1、上传到网站根目录 2、用phpMyad…...
机器学习中的神经网络重难点!纯干货(上篇)
. . . . . . . . .纯干货 . . . . . . 目录 前馈神经网络 基本原理 公式解释 一个示例 卷积神经网络 基本原理 公式解释 一个示例 循环神经网络 基本原理 公式解释 一个案例 长短时记忆网络 基本原理 公式解释 一个示例 自注意力模型 基本原理…...
[DDR4] DDR1 ~ DDR4 发展史导论
依公知及经验整理,原创保护,禁止转载。 专栏 《深入理解DDR4》 内存和硬盘是电脑的左膀右臂, 挑起存储的大梁。因为内存的存取速度超凡地快, 但内存上的数据掉电又会丢失,一直其中缓存的作用,就像是我们的工…...
享元和代理模式
文章目录 享元模式1.引出享元模式1.展示网站项目需求2.传统方案解决3.问题分析 2.享元模式1.基本介绍2.原理类图3.外部状态和内部状态4.类图5.代码实现1.AbsWebSite.java 抽象的网站2.ConcreteWebSite.java 具体的网站,type属性是内部状态3.WebSiteFactory.java 网站…...
[英语单词] ellipsize,动词化后缀 -ize
openvswitch manual里的一句话:里面有使用ellipsize,但是查字典是没有这个单词,这就是创造出来的动词。将单词ellipsis,加动词化后缀,-ize。 Often we ellipsize arguments not important to the discussion, e.g.: &…...
自然资源-测绘地信专业术语,值得收藏!
自然资源-测绘地信专业术语,值得收藏! 1、1954年北京坐标系 1954年我国决定采用的国家大地坐标系,实质上是由原苏联普尔科沃为原点的1942年坐标系的延伸。 2、1956年黄海高程系统 根据青岛验潮站1950年一1956年的验潮资料计算确定的平均海面…...
如何在小程序中实现页面之间的返回
在小程序中实现页面之间的返回,通常有以下几种方法,这些方法各有特点,适用于不同的场景: 1. 使用wx.navigateBack方法 描述:wx.navigateBack是微信小程序中用于关闭当前页面,返回上一页面或多级页面的API…...
深入解析数据结构之B树:平衡树中的王者
在计算机科学中,数据结构是算法和程序设计的基础。而在众多数据结构中,B树作为一种平衡树,在数据库和文件系统中有着广泛应用。本文将详细介绍B树的概念、特点、操作、优缺点及其应用场景,帮助读者深入理解这一重要的数据结构。 …...
18. 第十八章 继承
18. 继承 和面向对象编程最常相关的语言特性就是继承(inheritance). 继承值得是根据一个现有的类型, 定义一个修改版本的新类的能力. 本章中我会使用几个类来表达扑克牌, 牌组以及扑克牌性, 用于展示继承特性.如果你不玩扑克, 可以在http://wikipedia.org/wiki/Poker里阅读相关…...
OperationalError: (_mysql_exceptions.OperationalError)
OperationalError: (_mysql_exceptions.OperationalError) (2006, MySQL server has gone away) 这个错误通常表示客户端(例如你的 Python 程序使用 SQLAlchemy 连接到 MySQL 数据库)和 MySQL 服务器之间的连接被异常关闭了。这个问题可能由多种原因引起,以下是一些常见的原…...
DocGraph相关概念
结合简化版的直观性和专业版的深度,我们可以得到一个既易于理解又包含专业细节的DocGraph概念讲解。 DocGraph概述(简化版) 想象DocGraph就像是文章信息的地图。它通过拆分文档、识别关键词、分析关系,并最终以图形方式呈现这些…...
MySQL限制登陆失败次数配置
目录 一、限制登陆策略 1、Windows 2、Linux 一、限制登陆策略 1、Windows 1)安装插件 登录MySQL数据库 mysql -u root -p 执行命令安装插件 #限制登陆失败次数插件 install plugin CONNECTION_CONTROL soname connection_control.dll;install plugin CO…...
洛谷题解 - P1192 台阶问题
目录 题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1 提示代码 题目描述 有 N N N 级台阶,你一开始在底部,每次可以向上迈 1 ∼ K 1\sim K 1∼K 级台阶,问到达第 N N N 级台阶有多少种不同方式。 输入格式 两个正整数 N , K …...
Unity贪吃蛇改编【详细版】
Big and small greedy snakes 游戏概述 游戏亮点 通过对称的美感,设置两条贪吃蛇吧,其中一条加倍成长以及加倍减少,另一条正常成长以及减少,最终实现两条蛇对整个界面的霸占效果。 过程中不断记录两条蛇的得分情况,…...
React中数据响应式原理
React作为当下最流行的前端框架之一,以其声明式编程和组件化架构而广受开发者喜爱。而React的数据响应式原理,是其高效更新DOM的核心机制。本文将深入探讨React中数据响应式原理,并结合代码示例进行论证。 响应式原理概述 在React中&#x…...
【FreeRTOS】ARM架构汇编实例
目录 ARM架构简明教程1. ARM架构电脑的组成1.2 RISC1.2 提出问题1.3 CPU内部寄存器1.4 汇编指令 2. C函数的反汇编 学习视频 【FreeRTOS入门与工程实践 --由浅入深带你学习FreeRTOS(FreeRTOS教程 基于STM32,以实际项目为导向)】 https://www.…...
【Linux】常见指令的使用
文章目录 which指令stat 指令wc指令echo指令tree 指令whoami指令clear指令alias指令ls指令pwd指令cd 指令touch指令mkdir指令(重要)rmdir指令 && rm 指令(重要)man指令(重要)cp指令(重要…...
C#面:详细阐述什么是 DTO
DTO(Data Transfer Object)是一种设计模式,用于在不同层之间传输数据。它的主要目的是在应用程序的不同部分之间传递数据,而不是直接传递实体对象。DTO通常是一个简单的POCO(Plain Old CLR Object)…...
「TCP 重要机制」三次握手四次挥手
🎇个人主页:Ice_Sugar_7 🎇所属专栏:计网 🎇欢迎点赞收藏加关注哦! 三次握手&四次挥手 🍉连接管理🍌三次握手🍌意义🍌四次挥手🍌TCP 状态转换…...
Java数据库编程
引言 在现代应用开发中,与数据库交互是不可或缺的一部分。Java提供了JDBC(Java Database Connectivity) API,允许开发者方便地连接到数据库并执行SQL操作。本文将详细介绍Java数据库编程的基础知识,包括JDBC的基本概念…...
决策树算法介绍:原理与案例实现
一、引言 决策树是一种常用于分类和回归任务的机器学习算法,因其易于理解和解释的特点,在数据分析和挖掘领域有着广泛应用。本文将介绍决策树算法的基本原理,并通过一个具体案例展示如何实现和应用该算法。 二、决策树算法原理 1. 决策树结…...
业务代表模式
业务代表模式 引言 在软件工程中,设计模式是解决常见问题的经典解决方案。它们为开发人员提供了一种方法,以优雅和可重用的方式处理软件开发中的挑战。业务代表模式(Business Delegate Pattern)是一种行为设计模式,它主要关注于将业务逻辑与表示层(如用户界面)分离,以…...
LeetCode 算法:反转链表 c++
原题链接🔗:反转链表 难度:简单⭐️ 题目 给你单链表的头节点 head ,请你反转链表,并返回反转后的链表。 示例 1: 输入:head [1,2,3,4,5] 输出:[5,4,3,2,1] 示例 2:…...
【多线程】Thread类及其基本用法
🥰🥰🥰来都来了,不妨点个关注叭! 👉博客主页:欢迎各位大佬!👈 文章目录 1. Java中多线程编程1.1 操作系统线程与Java线程1.2 简单使用多线程1.2.1 初步创建新线程代码1.2.2 理解每个…...
Springboot 整合 Flowable(一):使用 flowable-UI 绘制流程图
目录 一、Flowable简介 二、Flowable 与 Activiti 的区别 三、流程图的绘制(以员工请假流程图为例) 1、下载 flowable 的压缩包: 2、启动包中的 tomcat 3、登录页面 4、绘制结束,导出 bpmn20.xml文件 一、Flowable简介 Fl…...
课设--学生成绩管理系统(一)
欢迎来到 Papicatch的博客 文章目录 🍉技术核心 🍉引言 🍈标识 🍈背景 🍈项目概述 🍈 文档概述 🍉可行性分析的前提 🍈项目的要求 🍈项目的目标 🍈…...
thinkphp5模型的高级应用
ThinkPHP5 是一个基于 PHP 的轻量级框架,它提供了许多便利的功能来简化 Web 开发。在 ThinkPHP5 中,模型(Model)是 MVC(Model-View-Controller)架构中的重要组成部分,负责处理数据逻辑。以下是一…...
网站开发有限公司/百度首页 百度一下
向set集合中插入元素时,可根据set.insert().second的返回值判断集合中是否已有该元素。 #include<set> using namespace std; int main() {set<char>a;a.insert(a);if(a.insert(a).second){printf("插入成功");}else{printf("插入失败&…...
南通门户网站建设/超级外链工具有用吗
NOTE: 1.APIs往往要求访问原始资源(raw resources),所以每一个RAII class应该提供一个“取得其所管理之资源”的办法。 2.对原始资源的访问可能经由显示转换或隐式转换。一般而言显示转换比较安全,但隐式转换对客户比较方便。 转载于:https:/…...
工作室网站建设费用价格/产品销售方案与营销策略
vue3生命周期、和hookvue3生命周期图生命钩子(与name、setup配置项同级)组合api形式写法hook位置:内容:使用:总结vue3生命周期图 生命钩子(与name、setup配置项同级) beforeCreate() {console.log("-----beforeCreate-----");},created() {con…...
企业网上推广方式/windows优化大师免费
LeetCode 1401. 圆和矩形是否有重叠 难度 中等 给你一个以 (radius, x_center, y_center) 表示的圆和一个与坐标轴平行的矩形 (x1, y1, x2, y2),其中 (x1, y1) 是矩形左下角的坐标,(x2, y2) 是右上角的坐标。 如果圆和矩形有重叠的部分,请…...
武汉建筑网站/知乎seo
C 二维数组动态分配和释放(1)已知第二维Code-1 char (*a)[N];//指向数组的指针a (char (*)[N])malloc(sizeof(char *) * m);printf("%d\n", sizeof(a));//4,指针printf("%d\n", sizeof(a[0]));//N,一维数组free(a);(2)已知第一维Co…...
推介做界面的网站/杭州seo推广优化公司
原文地址:http://kuangbaoxu.javaeye.com/blog/193076 1. 查询整个映射对象所有字段 //直接from查询出来的是一个映射对象,即:查询整个映射对象所有字段 String hql "from Users"; Query query session.createQuery(hql); List&…...