Unity3D手游开发日记(2) - 技能系统架构设计

我想把技能做的比较牛逼,所以项目一开始我就在思考,是否需要一个灵活自由的技能系统架构设计,传统的技能设计,做法都是填excel表,技能需要什么,都填表里,很死板,比如有的技能只需要1个特效,有的要10个,那么表格也得预留10个特效的字段.在代码里面也是写死一些东西,要增加和修改,就得改核心代码,如果我要把核心部分做成库封装起来,就很麻烦了.

能不能做成数据驱动的方式呢?

改技能文件就行了,即使要增加功能,也只需要扩展外部代码,而不用改核心代码,

我是这么来抽象一个技能的,技能由一堆触发器组成,比如特效触发器,动作触发器,声音触发器,摄像机震动触发器等等,这些触发器到了某个条件就执行触发,触发条件一般是时间,如果有比较复杂的浮空技能,可以增加落地触发等.

自定义一个技能文件,代替excel表格,看起来是这样:

简单的技能:

每一行都是一个触发器,这些触发器,到了某个条件会自动触发.

上面的意思就是,第0秒开始面向目标,第0秒开始播放动作1000

复杂的技能:

这个技能能将目标打到空中,并完成3连击,然后从空中砸向地面,

CurveMove(0, 0.413, 104, 0, 0, 0, 0);的意思就是,第0.413秒,开始做曲线运动,让角色飞到空中,曲线运动的ID是104,

用这样的文件来配置一个技能,很灵活,也很快,

[csharp] view plain copy

  1. private bool ParseScript(string filename)
  2. {
  3. bool ret = false;
  4. try
  5. {
  6. StreamReader sr = FileReaderProxy.ReadFile(filename);
  7. if (sr != null)
  8. ret = LoadScriptFromStream(sr);
  9. }
  10. catch (Exception e)
  11. {
  12. string err = "Exception:" + e.Message + "\n" + e.StackTrace + "\n";
  13. LogSystem.ErrorLog(err);
  14. }
  15. return ret;
  16. }
  17. private bool LoadScriptFromStream(StreamReader sr)
  18. {
  19. bool bracket = false;
  20. SkillInstance skill = null;
  21. do
  22. {
  23. string line = sr.ReadLine();
  24. if (line == null)
  25. break;
  26. line = line.Trim();
  27. if (line.StartsWith("//") || line == "")
  28. continue;
  29. if (line.StartsWith("skill"))
  30. {
  31. int start = line.IndexOf("(");
  32. int end = line.IndexOf(")");
  33. if (start == -1 || end == -1)
  34. LogSystem.ErrorLog("ParseScript Error, start == -1 || end == -1  {0}", line);
  35. int length = end - start - 1;
  36. if (length <= 0)
  37. {
  38. LogSystem.ErrorLog("ParseScript Error, length <= 1, {0}", line);
  39. return false;
  40. }
  41. string args = line.Substring(start + 1, length);
  42. int skillId = (int)Convert.ChangeType(args, typeof(int));
  43. skill = new SkillInstance();
  44. AddSkillInstanceToPool(skillId, skill, true);
  45. }
  46. else if (line.StartsWith("{"))
  47. {
  48. bracket = true;
  49. }
  50. else if (line.StartsWith("}"))
  51. {
  52. bracket = false;
  53. // 按时间排序
  54. skill.m_SkillTrigers.Sort((left, right) =>
  55. {
  56. if (left.GetStartTime() > right.GetStartTime())
  57. {
  58. return -1;
  59. }
  60. else if (left.GetStartTime() == right.GetStartTime())
  61. {
  62. return 0;
  63. }
  64. else
  65. {
  66. return 1;
  67. }
  68. });
  69. }
  70. else
  71. {
  72. // 解析trigger
  73. if (skill != null && bracket == true)
  74. {
  75. int start = line.IndexOf("(");
  76. int end = line.IndexOf(")");
  77. if (start == -1 || end == -1)
  78. LogSystem.ErrorLog("ParseScript Error, {0}", line);
  79. int length = end - start - 1;
  80. if (length <= 0)
  81. {
  82. LogSystem.ErrorLog("ParseScript Error, length <= 1, {0}", line);
  83. return false;
  84. }
  85. string type = line.Substring(0, start);
  86. string args = line.Substring(start + 1, length);
  87. args = args.Replace(" ", "");
  88. ISkillTrigger trigger = SkillTriggerMgr.Instance.CreateTrigger(type, args);
  89. if (trigger != null)
  90. {
  91. skill.m_SkillTrigers.Add(trigger);
  92. }
  93. }
  94. }
  95. } while (true);
  96. return true;
  97. }

文件的解析,也很简单

那么从代码上怎么实现呢?

1.触发器:

从同一个基类继承,

2.工厂模式来创建注册触发器,

在外部注册触发器的代码:

3.技能实例来管理触发器,

执行触发其实也可以写这里.

4.技能系统来管理所有技能

技能是可以复用的,技能系统就是一个技能池子,不停地new技能实例和回收技能实例

部分Public 接口代码:

总结一下思路,就是

SkillSystem 管理SkillInstance,创建和回收所有技能

SkillInstance 管理 SkillTrigger,负责触发器的触发.

SkillTrigger 就执行具体的效果.

代码封装上,可以把核心代码做成库,只开放触发器的扩展接口,项目已经在使用,很不错.

时间: 2024-09-30 14:37:10

Unity3D手游开发日记(2) - 技能系统架构设计的相关文章

Unity3D手游开发日记(6) - 适合移动平台的水深处理

市面上大部分的手机游戏,水面都比较粗糙,也基本没发现谁做过水深的处理. 水深的处理在PC平台比较容易,因为很容易获得每个像素的深度,比如G-Buffer,有了像素的深度,就能计算出每个像素到水面的距离,实现水深alpha渐变. 但是在移动平台,又是万恶的浮点纹理...导致此方案不行. 但是方案都是人想出来的,我想了两种适合移动平台的方案 方案1:用水面顶点颜色保存alpha值来做水深渐变. 这种方案,要求水面的模型面片是格子的,就像地形网格一样,格子越密,alpha的精度才越高. 方案2:用贴图

Unity3D手游开发日记(9) - 互动草的效果

所谓互动草,就是角色跑动或者释放技能,能影响草的摆动方向和幅度. 前面的文章早已经实现了风吹草动的效果,迟迟没有在Unity上面做互动草,是因为以前我在端游项目做过一套太过于牛逼的方案.在CE3的互动草的基础上扩展,效果好,但技术太复杂,效率开销也特别高. 如果在手机上,就得做一套简单高效的. 实现效果:从任意方向碰一下草,草就应该来回晃动,晃动幅度逐渐减小.多次触碰,效果应该叠加.这样的话就比较真实. 实现原理:用正玄波实现草来回摆动的简谐运动,用指数衰减来模拟阻力 实现步骤: 1.每个草挂一

Unity3D手游开发日记(4) - 适合移动平台的热浪扭曲

热浪扭曲效果的实现,分两部分,一是抓图,二是扭曲扰动.其中难点在于抓图的处理,网上的解决方案有两种,在移动平台都有很多问题,只好自己实现了一种新的方案,效果还不错. 网上方案1. 用GrabPass抓图 GrabPass在有的手机上是不支持的...效率也是问题,所以... 代码可以看看: [csharp] view plain copy Shader "Luoyinan/Distortion/HeatDistortion" { Properties { _NoiseTex ("

Unity3D手游开发日记(1) - 移动平台实时阴影方案

阴影这个东西,说来就话长了,很多年前人们就开始研究出各种阴影技术,但都存在各种瑕疵和问题,直到近几年出现了PSSM,也就是CE3的CSM,阴影技术才算有个比较完美的解决方案.Unity自带的实时阴影,也是这种技术,在电脑上很成熟的.我也是目睹了阴影技术的发展,以前也自己写过一些阴影,本来以为以后再也不用担心阴影了,结果做移动游戏以后,发现情况没那么乐观.这种基于深度的shadow map 技术,需要浮点纹理的支持,但有的移动平台不支持浮点纹理,即使支持,这种技术开销也特别大.所以我们需要重新思考

Unity3D手游开发日记(7) - 适合移动平台的天气效果

腾讯的天涯明月刀的天气很棒,以前我也在CE3的基础上做了一个效果差不多的,但是在手机上,还是斜下固定视角的游戏,是否还需要一个天气系统? 而且没有G-Buffer的支持,很多牛逼效果实现不了,比如角色身上的湿润感,水面的波纹涟漪等 考虑了一下,觉得可以做一个简单高效的. 以下雨天为例子.一阵风吹来,天色逐渐变暗,突然下起了雨来,雨越来越大,夹杂着雷声和闪电,过几分钟,雨逐渐消失... 1.风 前面文章已经实现了风引起植被的摆动 2.天色 这个比较麻烦,由于移动平台的场景用的基本都是lightma

Unity3D手游开发日记(5) - 适合移动平台的植被随风摆动

一直在思考怎么让场景更有生机,我觉得植被的随风摆动是必不可少的.CE3引擎的植被bending就做得特别棒.我也准备在手机上做一套. 先分析一下植被摆动常见的几种做法.其实不管哪种做法,核心就是让植被顶点做动画,有的顶点动的少(比如树根),有的顶点动的多(比如树顶),根据怎么样的权重来动? 方案1:  用UV来做权重. 这种方案对UV展开有要求,要从0到1,只适合面片草,这样的话草的根部和顶部的摆动权重就是一个0到1的线性的变化,随便用一个正玄波就能实现简单摆动了, [csharp] view

Unity3D手游开发日记(3) - 场景加载进度条的完美方案

我以为做个进度条很简单,分分钟解决,结果折腾了一天才搞定,Unity有很多坑,要做完美需要逐一解决. 问题1:最简单的方法不能实现100%的进度 用最简单的方法来实现,不能实现100%的进度,原因是Unity加载完新场景立马就激活新场景了,无法显示最后的进度.解决办法就是使用allowSceneActivation来控制进入场景的时机. 问题2:使用allowSceneActivation后进度卡在90% 这个问题官网论坛也有人讨论,解决办法就是自己手动修补最后的10%, 问题3:进度条一顿一顿

Unity3D手游开发日记(8) - 运动残影效果

2D游戏的残影很简单,美术做序列帧图片就行了,那么3D游戏的残影美术做不了,得靠程序员动态创建模型来处理. 实现原理也很简单: 1.间隔一定时间创建一个残影模型 [csharp] view plain copy GameObject go = GameObject.Instantiate(origin, pos, dir) as GameObject; 2.对残影模型采用特殊的shader,要简单高效 [csharp] view plain copy public class MotionGho

Unity3D手游开发实践

<腾讯桌球:客户端总结> 本次分享总结,起源于腾讯桌球项目,但是不仅仅限于项目本身.虽然基于Unity3D,很多东西同样适用于Cocos.本文从以下10大点进行阐述: 架构设计 原生插件/平台交互 版本与补丁 用脚本,还是不用?这是一个问题 资源管理 性能优化 异常与Crash 适配与兼容 调试及开发工具 项目运营 ? 1.架构设计 好的架构利用大规模项目的多人团队开发和代码管理,也利用查找错误和后期维护. 框架的选择:需要根据团队.项目来进行选择,没有最好的框架,只有最合适的框架. 框架的使