日志 教程
当前位置: 教程  >  游戏开发  >  手机游戏开发  >  正文

《刀塔传奇》主程张振新:游戏开发客户端的技术经验分享

作者:云世界NPC 发表于 2015/8/27 8:44:54     评论(5)     阅读(4865)     

《刀塔传奇》选择Cocos2d-x结合小巧高效的Lua,提高开发效率,对手游面临的复杂版本更新问题也颇为给力。

汇众教育联合游戏行业领军企业研发的《跨平台游戏程序开发专业》,也会完整教授相关技术课程。


以下整理自2014 Cocos开发大会,程序员张振新的演讲内容:


大家好,我叫张振新,来自莉莉丝科技,我从2009年开始负责PC端游的开发,2013年非常荣幸的进入莉莉丝科技,参与了《刀塔传奇》的开发,这是我开发的第一款手游。


下面为大家分享一下《刀塔传奇》客户端技术经验。


首先介绍《刀塔传奇》的一个客户端的整个结构。

我们在Cocos2d-x下面分装了一个数据模块,一个网络模块,最大的一块是业务逻辑模块,里面分UI系统、战斗系统和事件系统,大家玩的游戏都是通过这些模块搭建起来。



我们这个模块除了Cocos2dx是在C++层面实现的,《刀塔传奇》很大程度上都是通过Lua实现的。


开发效率

Lua是一个小巧的成本语言,它的设计目的是为了嵌入应用程序中,从而为应用程序提供灵活的扩展和定制功能。它有几个特点:

一、轻量级,官方版本只提供了一个精简的核心和最剧本的库,因此Lua体积小,启动速度快,因此非常适合嵌入在别的程序里。

二、可扩展性
,Lua并不象其它许多"大而全"的语言那样,包括很多功能,比如网络通讯、图形界面等。但是Lua提供了非常易于使用的扩展接口和机制:由宿主语言(通常是C或C++)提供这些功能。

三、非常容易上手,而且简单灵活
,这一点对用户有帮助,因为它学习比较方便,操作灵活。


Lua是一门解释性语言,没有编译和链接生产二进制代码的过程,它是由lua虚拟机直接解释执行的。开发中,修改完代码后,直接运行程序即可看见效果。因此可以节省大量的开发时间,项目越大,效果越明显。


刀塔传奇支持游戏过程中动态加载lua代码,省去重新打开程序的时间。我们在脚本层封装了一个reload函数,用于在游戏运行时重新加载lua代码。

实现原理也非常简单:

先将先前required过的module从删除,然后重新require该所有module,相当于重新初始化脚本代码。这样省去了关闭和重新打开程序的时间。

刀塔传奇的UI场景是使用cocos2dx的场景scene,并在脚本层维护了一个scene的stack,reload的时候将原先的scene先pop,然后使用reload后的代码和资源重新创建新的scene,还原原先的场景,省去了重复操作的时间。


版本更新

除了开发效率,Lua给我们《刀塔传奇》带来的好处就是版本更新。

版本更新是手游上市之后的非常重要的环节,直接影响到手游产品的质量。

手游版本的更新比PC更加的恶劣,因为iOS的AppStore有一段忽长忽短的审核期,而Andriod更糟糕,渠道众多,版本分发流程复杂。

刀塔传奇在立项的阶段就考虑到版本更新的问题,程序的整个框架也给版本更新带来一些便利性。

1) 线上热更

loadstring:功能非常简单,效果非常明显。先前做PC游戏的时候,每次发现线上紧急问题,采取的办法都是紧急停服,然后出更新包,测试,发布,周期长,体验差。曾经无数次线上紧急问题都是通过这个方式修复。

刀塔传奇在程序启动的时候,会从服务器上面拉一段代码下来,然后执行这段代码。由于这个逻辑是在发生在lua代码加载之后,于是就可以用拉取下来的代码段覆盖版本里面的lua代码。

注意:用loadstring只能覆盖全局域下面能访问到的函数,从程序设计方面来说,一个模块应该对外暴露尽可能少的接口,但是从线上热更来看,应该让全局域下面能访问到尽可能多的函数,这样可以提高线上热更代码的覆盖率。

2)游戏内更新包

这是刀塔传奇的常规更新方式,一般用来更新大特性版本。更新的内容包括lua脚本,美术资源,策划资源。由于代码都是lua脚本,更新完成后,可以通过重新初始化整个lua state,达到更新包在游戏运行时生效。

注意:更新包不能用于解决更新流程之前的bug;当一个版本有连续几个比较大的更新包之后,可以考虑出一个非强制更新的全量包给到渠道,这样有助于提高新进玩家的转化率。

3)全量包更新

这个是我们最不愿意采用的版本更新方式,有一些时候不得不用。比如《刀塔传奇》刚刚上线的时候跨服蓝牙堆栈功能。

这边每个数据跟大家分享一下,每次完整包更新的时候,当日的日活跃大概会少10多万。

它的优化方案,制作低清版本,解决包大小的问题,还有一种是在游戏内准备继承一个下载器的功能,就是在游戏启动过程当中,跟游戏更新包类似,在游戏启动过程当中完成一个全量包的下载和安装,达到优化更新体验的目的。


调试问题

Lua的开发者经常会碰到的感觉,就是开发效率极快,一下子做起来了,但是维护特别的困难,其中很重要的原因就是调试困难,很难找到bug。

除了常规的print和log的方法,刀塔传奇使用lua的debug库开发了一个DEBUG函数,作用类似ide环境下面的断点。该函数会先输出当前的调用堆栈,并进入循环等待用户输入调试语句,在该状态下,通过debug库提供的getinfo等函数可查看指定level堆栈下的local和upvalue。


错误提示框

我们还做了一个游戏错误框,类似这样的效果图。





《刀塔传奇》是在PC下面开发的,等版本发布出来,到Andriod、iOS一跑就各种问题,也不知道出现了什么问题,然后到各种下面去查非常的麻烦。

然后我们就在手机发布上面去添加了这么一个框,当有错误的时候它可以以这个形式展现过来,我们在发布内部的测试版方面,我们发这么一个框,可以很容易的定位到手机版本的问题,他们提交的bug单可以让我们找到错误的信息,这对bug来说是很方便的事情。


Snapshot:解决Lua脚本内存泄漏

使用lua开发也会存在内存泄漏的问题,snapshot可以对当前的Lua State做一个完整的快照,并记录对象的引用关系,我们可以在不同的手机端对Lua进行两次快照,通过两次的对比可以的出新增加的内存处于何处。

它在第一行做了一个快照,分配了一个Test1,然后他又做了一个S2,这对我们定位的内部发生了斜路都有很好的解决。特别是对Lua问题发生的不太容易发现的,通过添加这种方式可以解决。


性能问题

把性能瓶颈相关的代码用C或C++实现。

为了解决draglist渲染效率低下的问题,添加了可见性裁剪的功能,将不在显示区域的item项设置为不可见。第一版的做法是通过通过lua导出接口,在lua层实现裁剪,确实减少了draw call的调用,不过却增加了cpu的负担,主要是因为产生了大量的lua调用导出接口原因。第二版就直接在node层面上现实自动裁剪,在visit的函数里面判断是否可见。

对table预先分配大小,减少rehash。

当我们把新的键值赋给table的时候,若数组和哈希表已经满了,更会触发一个再哈希(rehash),再哈希的代价是高昂的,首先会在内存中分配一个新的长度的数组,然后将所有记录再全部哈希一遍,将原来的记录转移到新数组中。

Lua的创新跟其他的脚本实现方式不太一样,它采用了内例化的实现方式,所有的字符串在Lua中都只储存一份拷贝,并且都以引用的方式保存。这边可以考虑一下,Lua保存的方式可以用table保存。

 

评论
显示
悄悄话
    熊航
  • 熊航的评论:
  •   深奥
    2017-05-14 22:57
    熊航
  • 熊航的评论:
  •   深奥
    2017-05-14 22:57
    郭飞
  • 郭飞的评论:
  •   赞!
    2015-10-24 15:18
    赵熙猛
    2015-10-24 14:57
    姚瑶
  • 姚瑶的评论:
  •   hao
    2015-09-14 14:06
汇众教育官网 | 联系方式 | 版权声明 | 友情链接
Copyright 2008© 汇众益智(北京)教育科技有限公司. All Rights Reserved
京ICP备09092043号 京公网安备11010802009023号