最近有位同学面试Cocos Creator,我们把面试时问道的真题列举出来,并配上参考答案。
问题1: 你们公司项目时如何做战斗系统的?
面试官你好,做战斗系统和架构的时候,我们一般把代码逻辑分成3层来设计,同时把数据独立出来,我们的项目设计如下,不同公司可能不一样,我们之前公司是这样做的:
1: 我们会把战斗中的功能,做成组件,比如,动画切换,伤害计算等; 我们叫做功能层;
2: 我们会把战斗单元做一个组件,比如Charactor, 然后让Charactor来has 功能组件,并在Charactor中提供标准的通用的策略,比如,行走,技能释放等; 同时不是的角色,我们会编写Player, Enemy等来继承自Charactor,并绑定对应角色的战斗数据,用战斗数据来驱动统一的逻辑;
3: 操作层:我们会把战斗中的操作独立出来,比如UI操作,网络事件,AI事件等,通过事件模块,来调用到战斗中的具体策略,这样,可以实现不同操作层来驱动战斗,调用战斗策略;
4: 战斗中的数据,我们一般都用excel表格来给策划填写,更具策划的游戏需求来制定好对应的数据格式。如果有必要,还可以做一些可视化工具(技能编辑器),来给策划使用,最后导出战斗数据。
5: 技能编辑器:可视化的制定技能的一些数据,比如buff时间,攻击时间,等具体要根据游戏需求来定;
对惹,这里有一 个游戏开发交流小组,希望大家可以点击进来一起交流一下开发经验呀!
问题2: 项目中出现多个spine卡顿怎么办?
目前cocos creator 提供spine的动画组件,他有几种模式,性能最好的是预先Bake模式,就是把每个骨骼在某一帧的位置预先计算出来然后存起来,这样能提升性能。可以开启spine动画的合批模式,降低Drawcall,但是有限制,需要每个spine只有一种混合模式,一张贴图才能进行Drawcall合并。如果多个spine还是卡顿,那么就把spine换成帧动画,提升一些性能。
问题3: 百人同屏如何优化drawcall?
如果是2D游戏,只要保证角色的动画是同一个图集的帧动画(spine动画好像无法合批,待确认)就能最大限度地合批,同时把角色UI属性地相关内容,和原来的2D角色分开,把UI单独做,然后UI节点和角色节点做位置同步,这样,角色的动画就可以再一个图集合批,不会被自己的昵称,血条等打断合批。
如果是3D游戏,就要将3D动画的每个顶点的位置,Baker到一个纹理里面,然后使用GPU Instancing合批,来优化掉3D动画组件,从而合批,同样,UI等,都要单独独立开发,这样防止打乱合批。
问题4: 你们项目的场景切换是怎样做的?
目前主流的游戏开发中场景切换做法有3个,不同的公司可能不一样:
1:直接把场景做好,然后内容方场景里面,调用cocos API, 来直接加载与运行对应的场景;
2: 直接把场景做成场景资源,打入ab包中,来调用cocos API,从资源加载中加载场景,然后运行;
3: 直接只用一个场景,通过预制体来做相关的内容展示,不存在场景切换。
具体的选择,可以根据框架与项目来决定。比如,3D游戏中需要场景中做静态光照,我们就可以采用方法2,把场景看作是地图。比如简单的游戏就可以直接使用方法1 or方法3。
建议采用方法2, or 方法3。方法1最大的问题,是场景无法分ab包,全部都打入到main包里面,这样做分包的时候会有问题;
问题5: 动画的帧率是多少最为合适?
动画的帧率,要看,都可以,主要根据动画的类型来定,可以是15FPS, 30FPS, 60FPS.一般目前骨骼动画这些帧率都是控制在15~30FPS,有些可以是10FPS。具体要根据,帧率,资源大小等,做一个平衡,比如帧动画,帧过多,就会导致内存占用过大,帧过少,动画又不流畅。
问题6: Gpu Instance 怎么用的?
1: 要满足GPU instancing合批的条件,同一个材质,同一个shader,同一个mesh;
2: 渲染的时候,不能被别的物体打乱;
3: 在材质的地方勾选Gpu instancing选项,就可以实现Instancing 合批;
问题7: 你在游戏开发中是如何做内存管理得;
面试官你好,关于使用Cocos Creator开发做内存管理,可能需要注意得点,我是这样看的:
Cocos Creator开发游戏的内存占用主要是由以下部分组成,
1: 逻辑数据的内存占用,比如,节点,组件实例,字符串对象等;
2: 纹理,模型顶点等显存的内存占用;
3: 就是加载资源,场景切换过程中的一些中间状态的内存占用,比如,解码一个png的图片到内存,可能会要从资源里面把资源数据读到内存中再解码。
对于3的中间状态,最终会被引擎所释放,我们只要关注切换,加载时候的内存峰值,不要让内存峰值引发奔溃就可以了,如果内存峰值很高,这个时候,我们可以考虑
延后加载等。具体还要根据项目来定位。
对于2,我们一般是把不用的资源即时的给他释放掉,这个时候,我们做资源管理的时候,要尽量的把同一个功能场景的资源,打入一个ab包,这样,用的时候整个ab包都用,不用的时候都可以释放。
对于2,我们只要根据内存的峰值情况,再特定的时机,把不用的资源释放掉即可,如果我们的内存可以承受,那么可以不释放。
对于1, TypeScript/JavaScript 一般的都有垃圾回收的机制,我们一般不用考虑释放的问题,直接new 就可以了,但是也要避免大量的反复的new, 比如,在update里面不断地new 一个对象;
这种要尽量地避免。当我们地内存出现峰值地时候,可以主动地启动垃圾回收来进行回收处理。对于大量反复创建地节点,我们可以考虑使用缓存池来处理,这样避免直接像系统反复地new/free。
目前是我对Cocos Creator开发,关于内存要注意地点,可能有遗漏,具体项目还要根据实际情况具体分析。
问题8.你认为怎样的编码才算规范?
编码规范这个不同的项目,不同的公司可能要求不一样,我理解的规范的代码,大概从几个方面讲:
1: 用英文来做好命名,命名要能代表函数or变量or类的功能和作用,命名的习惯可以统一,比如
私有变量用_开头等;
2: 要注意权限,改私有的私有,改公有的公有;
3:命名规范的时候,可以和引擎做适当的一些区分;
4: 把逻辑流程的代码与具体功能代码来分开,流程代码负责流程,功能代码负责具体功能。
5: 写代码要符合项目的真实性,符合大家的习惯性认识,尽量少造概念等。
6:注释的一些要求,如果看具体项目和公司的团队成员的水平与协作来决定注释要写到什么程度
...
具体不同的项目规范不一样,还是可以提前规划好。