基于Quick-cocos2d-x的资源更新方案 二

写在前面


又是12点半了,对于一个程序员来说,这是一个黄金时间,精力旺盛,我想,是最适合整理和分享一些思路的时候了。

自从上次写了 基于Quick-cocos2d-x的资源更新方案

同样可见quick-cocos2d-x官方论坛贴子:http://www.cocoachina.com/bbs/read.php?tid=209421&fpage=2

一直觉得有些什么地方不对。思考了许久之后,发现对于framework和update自身模块的更新,是有一定的问题的。

对于update自身模块的更新来说,检查是否有更新,若有更新,则使用require "main"重新来过。 是非常合理。

而对于framework的更新,就不是那么容易了。 前一篇文章中,我提到若检查到framework有更新,则提示用户重启客户端,看似很不经意的操作,却给了用户思考要不要再次打开此游戏的机会。

对于这种思考机会,我们是不能给用户的。 因此,无缝更新才是硬道理。

BTW:目前我使用的是 quick-cocos2d-x 2.2.3rc 版本

 

改进后的热更新流程


在思索良久后,我发现对于update模块的实现,是完全可以脱离quick框架的。update模块所需要的scene,writablePath,md5file等功能,都不是quick提供的,仅仅是quick提供了一层封装。因此,我们可以使用纯cocos2d-lua的API来实现update模块。这样一来,framework.init就可以不在update模块里调用,那当我们在update中更新了framework模块时,我们完全不需要担心require重入问题。 因为在真正进入游戏前,我们的framework中的任何模块,都还没有被require过。

由于每天晚上只有1多小时的编程时间,所以,这事儿我又花了好几天。其间也遇上了不少问题。

我先来说说新版的更新流程

在前一文章的基础上,我们修改前三个步骤

1、从服务器取得update模块,与本地进行比较
2、检查update的md5值,看是否有更新,如果有更新,则下载update.bin,重新载入,并退到main(退出之前,注意清除对某些的引用),再次重新进入
3、如果本地update的值与服务器相同,表示update模块没有更新,则去下载文件列表 flist

.....后面的步骤不变

 

这样修改完毕后,我们就可以完全实现update模块的自更新,以及framework的自更新了。 整个过程不需要重新启动。

 

主要问题与解决办法


下面来说说这几天遇上的问题

1、在ANDROID上读取APK内文件的MD5值

由于CCCrypto::md5file是用fopen读取文件内容,再交给MD5计算器得出MD5值。 所以,在APK里是不可以的,后来,我把fopen改成了CCFileUtils::getFileData得到解决

2、flist是一个LUA文件,如果处于APK中,则无法使用dofile,如果是从网上下载,也无法使用dofile

为了保持兼容,在读取APK和执行网上下载来的flist时,我使用了loadstring(str)() 这样就解决了所有问题

3、递归创建文件和清空文件夹

一开始使用了os.execute来执行mkdir,rm等命令,但一但执行命令时控制台有输出,那整个LUA的LOG就乱了。 后来改为了lfs手工创建和遍历删除来解决。工作良好

4、compile_scripts.bat打包

我需要将update打一个包,main.lua不打包,其余的app和config.lua打一个包

由于compile_script.bat是按文件夹进行模块剔除的,所以,我只好调整以下结构

scripts

   +--launcher_src

       +--launcher.lua

   app

   +---MyApp

   +---config.lua

   +---scenes

 --main.lua

 

可以看出,我将更新模块单独放入了一个文件夹,将config.lua移到了app下面。 差点忘了说,我的update模块,也是没有依赖config.lua的。

这样,我就可以打出app和luancher两个包了,阳光七月的打包后缀是*.bin,我用了png,总之,这个后缀是无关紧要的,打包我写了两个脚本

build_app.bat

内容:%QUICK_COCOS2DX_ROOT%/bin/compile_scripts.bat -i scripts/app_src -p app -o res/app.png

build_lchr.bat

内容:%QUICK_COCOS2DX_ROOT%/bin/compile_scripts.bat -i scripts/lchr_src -p lchr -o res/lchr.png

 

这样当我们想要运行程序的时候,只需要打包一下就可以了。

 

F5实时刷新


考虑到很多时候,我们改了代码就想直接按F5,不想去打包……。 这里有两个方案。

一是处理一下PLAYER的F5事件,自动执行上面两个脚本。

二是处理一下main.lua,使它支持打包和非打包模式,目前,我是通过修改main.lua来做的

LOAD_FROM_BIN = true
APP_NAME = "dota"
APP_PACKAGE_ROOT = ""

 

if LOAD_FROM_BIN then
    APP_PACKAGE_ROOT = "app"
    CCLuaLoadChunksFromZIP("res/lchr.png")
    package.loaded["lchr.launcher"] = nil
    require("lchr.launcher")
else
    APP_PACKAGE_ROOT = "app_src"
    package.loaded["lchr_src.launcher"] = nil
    require("lchr_src.launcher")
end

 

让一切消散在风里


写到这里,已经不知道自己还要说些什么了。虽然目前功能不够完善,但已经可以发码了。希望能够给大家一个参考。

源码下载地址

 

posted @ 2014-06-28 01:09  麒麟子MrKylin  阅读(2209)  评论(2编辑  收藏  举报