运行环境

win7 64

cocos2d-2.1rc0-x-2.1.2

lua 5.1

通常我们编写好的lua代码都是明文形式,谁都可以查看修改,为了防止自己的劳动成果不被别人轻易的盗取,可以使用luac(lua库中自带)对其进行加密,转换为二进制文件。这样lua代码就无法直接查看,但是这里会有一个问题:在windows下能够很好的运行,在android上就会黑屏,提示错误:
[LUA ERROR] binary string: unexpected end in precompiled chunk

追根溯源

  在bool AppDelegate::applicationDidFinishLaunching()中查看lua加载代码:

view sourceprint?
01.#if (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID)
02.CCString* pstrFileContent = CCString::createWithContentsOfFile( "program/main.lua" );
03.if (pstrFileContent)
04.{
05.pEngine->executeString(pstrFileContent->getCString());
06.}
07.#else       
08.std::string path = CCFileUtils::sharedFileUtils()->fullPathForFilename( "program/main.lua" );
09.pEngine->addSearchPath( path.substr( 0, path.find_last_of( "/" ) ).c_str() );
10.pEngine->executeScriptFile( path.c_str() );
11.#endif

这里executeString()加载lua文件,继续查看源码发现真正干活的是:

view sourceprint?
1.LUALIB_API int luaL_loadbuffer (lua_State *L, const char *buff, size_t size,
2.const char *name) {
3.LoadS ls;
4.ls.s = buff;
5.ls.size = size;
6.return lua_load(L, getS, &ls, name);
7.}

在回头看看函数executeScriptFile()的执行过程:

view sourceprint?
01.LUALIB_API int luaL_loadfile (lua_State *L, const char *filename) {
02.LoadF lf;
03.int status, readstatus;
04.int c;
05.int fnameindex = lua_gettop(L) + 1 /* index of filename on the stack */
06.lf.extraline = 0;
07.if (filename == NULL) {
08.lua_pushliteral(L, "=stdin");
09.lf.f = stdin;
10.}
11.else {
12.lua_pushfstring(L, "@%s", filename);
13.lf.f = fopen(filename, "r");
14.if (lf.f == NULL) return errfile(L, "open", fnameindex);
15.}
16.c = getc(lf.f);
17.if (c == '#') {  /* Unix exec. file? */
18.lf.extraline = 1;
19.while ((c = getc(lf.f)) != EOF && c != '
20.') ;  /* skip first line */
21.if (c == '
22.') c = getc(lf.f);
23.}
24.if (c == LUA_SIGNATURE[0] && filename) {  /* binary file? */
25.lf.f = freopen(filename, "rb", lf.f);  /* reopen in binary mode */
26.if (lf.f == NULL) return errfile(L, "reopen", fnameindex);
27./* skip eventual `#!...' */
28.while ((c = getc(lf.f)) != EOF && c != LUA_SIGNATURE[0]) ;
29.lf.extraline = 0;
30.}
31.ungetc(c, lf.f);
32.status = lua_load(L, getF, &lf, lua_tostring(L, -1)); // 关键,与luaL_loadbuffer()中的一样
33.readstatus = ferror(lf.f);
34.if (filename) fclose(lf.f);  /* close file (even in case of errors) */
35.if (readstatus) {
36.lua_settop(L, fnameindex);  /* ignore results from `lua_load' */
37.return errfile(L, "read", fnameindex);
38.}
39.lua_remove(L, fnameindex);
40.return status;
41.}

注意看代码的中文注释部分,这里就能解释为什么用luac编译的文件在window下面可以好好地运行(IOS上面也OK),而android上就不能。luaL_loadfile()中通过c == LUA_SIGNATURE[0]判断lua文件是否加密,如果是,则重新以“rb”方式打开。

解决办法

修改scriptingluacocos2dx_supportCocos2dxLuaLoader.cpp中的loader_Android()函数

修改前:

view sourceprint?
01.int loader_<a href="http://www.it165.net/pro/ydad/" target="_blank" class="keylink">Android</a>(lua_State *L)
02.{
03.std::string filename(luaL_checkstring(L, 1));
04.filename.append(".lua");
05. 
06.CCString* pFileContent = CCString::createWithContentsOfFile(filename.c_str());
07. 
08.if (pFileContent)
09.{
10.if (luaL_loadstring(L, pFileContent->getCString()) != 0)
11.{
12.luaL_error(L, "error loading module %s from file %s :
13.%s",
14.lua_tostring(L, 1), filename.c_str(), lua_tostring(L, -1));
15.}
16.}
17.else
18.{
19.CCLog("can not get file data of %s", filename.c_str());
20.}
21. 
22.return 1;
23.}

修改后:

view sourceprint?
01.int loader_Android(lua_State* L)
02.{
03.unsigned char* pData = nullptr;
04.unsigned long size = 0;
05. 
06./* modify for read lua script from bytecode */
07.std::string filename(luaL_checkstring(L, 1));
08.std::string rel_name = filename.append(".lua");
09. 
10.pData = CCFileUtils::sharedFileUtils()->getFileData(rel_name.c_str(), "rb", &size);
11.if (pData) {
12.if (luaL_loadbuffer(L, (char*)pData,size,rel_name.c_str()) != 0 ) {
13.luaL_error(L, "error loading module s from file s :s",
14.lua_tostring(L, 1), rel_name.c_str(), lua_tostring(L, -1));
15.}
16.}
17.else {
18.CCLog("can not get file data of %s", filename.c_str());
19.}
20. 
21.CC_SAFE_DELETE_ARRAY(pData);
22. 
23.return 1;
24.}

另外,由于在cpp中第一次加载lua并没有调用loader_Android(),这是因为uaL_loadbuffer只会把文件加载成为一个chunk,而不会运行该chunk,所以还要在加一条调用语名lua_pcall,如下:

view sourceprint?
01.int loader_Android(lua_State* L)
02.{
03.unsigned char* pData = nullptr;
04.unsigned long size = 0;
05. 
06./* modify for read lua script from bytecode */
07.std::string filename(luaL_checkstring(L, 1));
08.std::string rel_name = filename.append(".lua");
09. 
10.pData = CCFileUtils::sharedFileUtils()->getFileData(rel_name.c_str(), "rb", &size);
11.if (pData) {
12.if (luaL_loadbuffer(L, (char*)pData,size,rel_name.c_str()) != 0 || lua_pcall(L, 0, LUA_MULTRET, 0) ) { // 修改处
13.luaL_error(L, "error loading module s from file s :s",
14.lua_tostring(L, 1), rel_name.c_str(), lua_tostring(L, -1));
15.}
16.}
17.else {
18.CCLog("can not get file data of %s", filename.c_str());
19.}
20. 
21.CC_SAFE_DELETE_ARRAY(pData);
22. 
23.return 1;
24.}

或者我们可以偷鸡一下,

view sourceprint?
1.pEngine->executeString("require "program/main""); // 注意这里的"

最后我们的AppDelegate::applicationDidFinishLaunching()大概就是这样的,

view sourceprint?
01.const char* luaFile = "program/main.lua";
02.#if (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID)
03.std::string path = CCFileUtils::sharedFileUtils()->fullPathForFilename(luaFile);
04.CCLog("path = %s", path.c_str());
05.std::string subPath = path.substr(0, path.find_last_of("/"));
06.CCLog("sub path = %s", subPath.c_str());
07. 
08.pEngine->addSearchPath(subPath.c_str());
09. 
10.std::vector<std::string> searchPaths = CCFileUtils::sharedFileUtils()->getSearchPaths();
11.searchPaths.insert(searchPaths.begin(), subPath);
12.CCFileUtils::sharedFileUtils()->setSearchPaths(searchPaths);
13. 
14.pEngine->executeString("require "program/main""); // 注意这里的"
15.#else       
16.std::string path = CCFileUtils::sharedFileUtils()->fullPathForFilename(luaFile);
17.pEngine->addSearchPath( path.substr( 0, path.find_last_of( "/" ) ).c_str() );
18.pEngine->executeScriptFile( path.c_str() );
19.#endif

注:这里笔者偷鸡了一下。

cocos2dxandroid运行Luac编译后的lua代码相关推荐

  1. cocos2dx android运行Luac编译后的lua代码

    环境: win7 64 cocos2d-2.1rc0-x-2.1.2 lua 5.1 通常我们编写好的lua代码都是明文形式,谁都可以查看修改,为了防止自己的劳动成果不被别人轻易的盗取,可以使用lua ...

  2. java jar反编译后保存_java根据jar包反编译后修改再打包回jar的做法

    1. 得到一个待要修改的jar包 2. 我的环境是windows,然后解压这个jar包,得到一堆class文件,这时候就找到你需要的那个class文件 3. 我首先是使用jd-gui工具看一下这个cl ...

  3. Linux环境下开发板Tiny4412应用,实现交叉编译及minicom的调配,将代码编译后下载到开发板并运行

    一.实验目的 1.熟悉Linux环境,学习使用命令行操控计算机系统,学会基础的ubuntu机器操作. 2.初步学习使用开发板Tiny4412,查看实验说明以及开发板说明书,学习基本使用步骤. 3.安装 ...

  4. cocos2d 3.3 lua 代码加密 luac

    1.0 cocos luacompile 用法 我用的普通的cocos2d lua,没用quick,quick好像可以对整个资源包括图像和音频都加密,打包成zip.我看了下luacompile 的 h ...

  5. java 0000 0001 0002 000a 000b_Java代码编译后的class文件

    还是熟悉的味道,还是最简单的代码. // Hello.java public class Hello { public static void main(String[] args) { System ...

  6. Angular Component代码和编译后生成的JavaScript代码

    从 TypeScript 转换为 JavaScript 在这里称为编译. 在这种情况下,compiling 并不意味着创建二进制代码. 对于这种翻译,使用术语 transpilation 而不是 co ...

  7. stm32 Keil编译后查看代码/内存占用空间,Flash/RAM占用大小,Code-Data,RO-Data,RW-Data,ZI-Data是什么含义

    1. 查看码Flash/内存RAM占用大小 工程编译后,双击上图中红框位置,会打开 .map文件,包含了各个函数和文件占用的空间大小和地址. 2. Code-Data,RO-Data,RW-Data, ...

  8. 深入理解:cmd下java命令启动JVM运行class文件时,可以自动识别不同编码编译后的class文件并加之运行

    cmd下java命令启动JVM运行class文件时,可以自动识别不同编码编译后的class文件并加之运行 总结如下: 一:程序源代码以GBK格式保存时: 二:程序以源代码UTF-8无BOM格式保存时: ...

  9. 易语言服务器调试输出为假,易语言判断、如果真、文本到整数比较时,调试时运行正常,编译后不正常...

    今天在用易语言帮一个客户写程序的时候,发言一个奇怪的问题,就是在用判断.如果真命令比较2个文本到整数后的值时,在调试运行的时候,一切正常,但是编译出来后,运行不正常,不符合逻辑,最终经过调试输出,发现 ...

最新文章

  1. css 中文字旋转,css
  2. Makefile中的变量和shell变量
  3. python configparse_python中ConfigParse模块的用法
  4. editActionsForRowAtIndexPath(iOS8) tableview编辑(删除、插入、移动)
  5. 打造极致体验:字节跳动亿级 DAU 背后的音视频技术最佳实践
  6. Android解决ScrollView中的滑动效果导致GestureDetector中的OnFling不能正常工作问题
  7. 智伴机器人安卓怎么设置个人热点_周五到!!宿舍断网怎么办,教你一招
  8. Golang 反射操作整理
  9. Opencv4.5.5 + Opencv4.5.5_contrib 图像拼接
  10. C语言文件读写(输入输出重定向)
  11. Formality形式化验证脚本范本
  12. android o bln-al10,bln-al10是什么手机型号 bln-al10是啥手机型号
  13. 天啦噜!我竟然见到了超级网红区块链本人
  14. 三相直流无刷电机驱动
  15. svg怎么转换成png格式?
  16. 锦上添花之元音音标学习(上)
  17. 达摩院预测的2023十大科技趋势,你看准哪个?
  18. 视频教程-wordpress建站教程之环境部署详解课程-PHP
  19. 清华计算机系收自考本科生吗,成人高考可以考清华吗,成年人怎么样可以上清华北大?...
  20. STM32F工程移植注意事项

热门文章

  1. gd32f470总结
  2. Echarts圆形占比图改良版
  3. 浅析YUV422模式
  4. webstorm js中文乱码
  5. python 嵩天课件_0.4嵩天老师python爬虫笔记整理week4
  6. nyoj 1036非洲小孩
  7. 数字化棋牌室 | 会员管理预约系统 | 棋牌室小程序
  8. 产品经理和项目经理的区别,读这一篇就够了!(史上最全总结)
  9. 网络钓鱼技术解析与安全防护措施
  10. DUX7.5大前端WordPress主题开源无限制版