Cocos2d+Lua 游戏开发中的技巧系列之一 Lua require的语法糖

2024-09-04 10:32

本文主要是介绍Cocos2d+Lua 游戏开发中的技巧系列之一 Lua require的语法糖,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

参考文档  --- https://blog.csdn.net/lovesmiles/article/details/80939240

笔记:

Lua/JS尾调用   http://www.ruanyifeng.com/blog/2015/04/tail-call.html
Lua的闭包详解   (和JS一吊样)

Lua中的元表与元方法(_index与_newIndex区别) https://blog.csdn.net/u013654125/article/details/100151633
qucik lua中require和import的区别    
2种区别在定义对象时. class()/local X={}   (前者用new 后者直接用)

Lua - table输出乱序 (如果元素不为键值对,则按顺序遍历元素 Or乱序)         Lua与C++接口对照    类声明继承

【译】从 JS 学习 Lua    
 Cocos2d-x Lua 1.环境搭建并开始项目 https://blog.csdn.net/zhouyongku/article/details/72891661

VS新建类/引用库(lib/dll)  table模拟OOP/cc.exports.      断点无效(整了一天 教训:仔细查Log/提示) <2020/3/13>

 studio与cocos版本不批配(CSB无法加载)  回调函数(使用this=self) print/f输出nil  

如何查对应接口tolua_usertype  无声明提前    <踩蚂蚁code:下方>

 

2020/3/12/

cocos2d-lua:新建项目环境搭建弄了好久 <博客https://www.jianshu.com/p/b00d518c649c>
 报错:    运行引擎下setUp.py文件, DOS窗口Log提示:ANDROID_SDK_ROOT/NDK_ROOT not found
 解决办法:  环境变量Path中要引用%NDK_ROOT%.   SDK都配了一直not found是因为:路径要到tools 泪奔

2020/3/14/
cocosStudio与cocos版本不批配 (导致CSB无法加载)  

回调函数self不是原来self(类似JS解决: 使用this=self)

常见bug: <a nil value>

sp =cc.Sprite:create();  self:addChild(sp);

这种情况不外户: 对象或方法为nil

OOP: 面向对象

一些笔记:


        作为一个中级即将要成为高级的游戏开发程序员,用Lua写了一二年游戏的你,require这个有什么好提的,天天在用,使用中还有什么技巧可言?

  这个require在使用过程中,真的还有一些技巧,当然,这个技巧并不是说在游戏运行效率上有惊天的变化,而仅仅是一个语法糖,不用对游戏性能也没有半毛钱的影响,但是用了,可能对游戏开发效率和阅读效率都有些帮助。

    先分享一下require在Lua源代码中的实现...嗯,算了,这一步跳过吧,require加载一个lua的过程很多博客已经说了,我也说得并不比别人好,还是直接进入主题吧。

通常,很多游戏的reuire是这样子用的:比如,某游戏的包背管理器的是在PackageMgr.lua里面实现的,这个文件在游戏的Res\script\lua\mgr\PackageMgr.lua里面。假如PackageMgr.lua里面是这样子写的,即这个PackageMgr.lua是直接返回的一个单例。

local PackageMgr={}--这里面是游戏逻辑return PackageMgr


那么,在其它地方引用这个单例时,最常见的一种代码的写法是这样子:

方法1

local packet = require("script.lua.mgr.PackageMgr")

将Res设置成lua的搜索根路径,在引用时,使用全路径,这样子的用法是比较经典的,它的问题唯一在:写的代码太多了,你要引用别人写的一个模块,还要知道它所在的路径,假如这个路径很长,码起来很累的,copy也会出错,而且加载出错,要到运行时才能知道没有这个路径。

为了解决拼写过长的问题,我们可以考虑另外一种解决方案:

方法2

我们将Res\script\lua\mgr\添加到游戏的搜索路径,这样,在使用时就可以

local packet = require("PackageMgr")

只需要一个PackageMgr模块名就能将它引用进来了,是不是省了很多字?嗯,这种写法也有一个问题:就是假如游戏的文件夹太多,这样子就要添加很多的搜索路径,搜索路径太多的话,搜索的效率自然会降低了,而且,太多的文件路径,怎么添加?手工添加还是脚本添加?新增的和删除的文件夹怎么维护也是个问题。

好了,我们既想获得方法2的简洁,又想获得方法1 的高效率,自然的就出现了第三种方法

方法3

我们定义一个新的require,为了区别require,我们就叫做GameRequire吧。

function GameRequire(luaFileName)local fullLuaFilePath = FullPath[luaFileName]if fullLuaFilePath thenreturn require(fullLuaFilePath)elsereturn require(luaFileName)end
end


阅读上面GameRequire的实现代码,首先,通过一个全局的FullPath表,将luaFileName转换成FullLuaFilePath,假如存在FullLuaFilePath,我们就用FullLuaFilePath 进行require,假如不存在,我们就用luaFileName进行require。这样,我们在使用时只要输入一个lua文件名,就能自动实现全路径的搜索加载啦。相信大家都已经看出这里面实现文件名转换成长路径逻辑的关键是FullPath这个表了,这个表具体是长这样子的:

FullPath ={Package="script.lua.mgr.PackageMgr",
MainUI ="script.lua.ui.MainUI",
....} 


可以看出这个FullPath是一个全局的变量表,这个表里面的内容非常简单,就是将所有的lua文件都转换成了长路径保存,通常这个FullPath.lua是用脚本来实现和维护的,哪个同学新增了一个lua文件,就跑一下维护脚本,更新一下这个FullPath.lua,这样子大家都可以用GameRequire("文件名")就可以引用新增的模块了。

写到这里,相信很多写了几年lua的同学都想喷我了,什么辣鸡啊,这些东西也有什么好写的,不是很常见的吗?没有什么好说的。好吧,好吧,很显然,写GameRequire还是太累了,虽然我们可以把GameRequire简写成GR之类的,但还要写引号,括号。下面的方法就是解决这个懒人问题的奇淫技巧。这种方法是我自己原创出来的(虽然技术含量基本没有,可能也有很多大佬不宵,不过我觉得还是很意思的)。

方法4

我们知道lua里面有元表这个东西,元表里面的__index方法相信很多同学都知道了,这里面我们要用__index方法来实现lua文件的加载,使得我们可以少写点代码

 

local GameRequire= GameRequire
_GModel ={}
_GModelMeta={
__index =function(t,key)local modelFile = GameRequire(key)--t[key] = modelFile --这里是开启缓存功能return modelFileend}


我们定义了一个全局的变量_GModel,这个变量有一个元表_GModelMeta,里面有一个元方法。使用时是这样子使用的:

local packet = _GModel.PackageMgr

这样子就实现require的调用了。原理听我慢慢分析一下:

首先: _GModel.PackageMgr和 _GModel["PackageMgr"]是等价的对吧?(这是lua的语法,如果你不知道的,那么这篇文章对你来说真的是白看了)。

其次:lua先在_GModel表里面查找“PackageMgr”成员,查不到,就在元表_GModelMeta里面查找,_GModelMeta里面也找不到啊,没关系,_GModelMeta有一个__index元方法,那就调用 __index方法来查找吧,这样子__index里面的GameRequire函数就被调用 了,然后返回GameRequire的结果。上面__index函数被调用里,里面的参数t就是_GModel,key就是“PackageMgr”,为什么这二个参数值是这样子?你得回去查查元表的调用过程了。

上面的代码还有一句注释掉了的,这个注释掉的是一个缓存代码,假如将它打开,第一次调用时_GModel找不到成员,会触发GameRequire,同时,将GameRequire的返回值存在了_GModel里面了,第二次调用时,因为_GModel已经有模块的数据,将会直接返回,不再调用GameRequire,也就是也不会调用到require了,省了lua内部的查找过程,在这个角度看,开了缓存的_GModel的效率比require还要高一点。写的代码比require少,效率还要比require高那么一点点的方法,是不是个奇技淫巧?

好了,到这里,我们就完成了require的奇技介绍了。如果你愿意你可以将_GModel简单写成_M

local packet = _M.PackageMgr

这样子,require是不是就可以码很少字啦?哈哈,不过千万不要将_GModel简单写成_G,因为_G是lua的保留变量。
————————————————
MainScene.lua
 

local MainScene = class("MainScene", cc.load("mvc").ViewBase);
local PlayScene = import("app.views.PlayScene");function MainScene:ctor()local sp = display.newSprite("img/cover.jpg");sp:setPosition(display.cx, display.cy);self:addChild(sp);local sp1 =  ccui.Button:create( "img/play_button.png","img/play_button.png","img/play_button.png" );sp1:setPosition(display.cx, display.cy - 200);self:addChild(sp1);sp1:addTouchEventListener(function(ref, evenType)if cc.EventCode.BEGAN == evenType thenelseif cc.EventCode.ENDED == evenType thenlocal playScene2 = PlayScene.new() ;print(" : ", display, display.replaceScene, import("app.views.PlayScene"), playScene2);cc.Director:getInstance():replaceScene(import("app.views.PlayScene").new()); -- display.replaceScene(playScene2)endend);
endfunction MainScene:onEnter()
endfunction MainScene:onExit()
endreturn MainScene

PlayScene.lua


local PlayScene = class("PlayScene", function()return display.newScene("PlayScene")
end)local Game = require("app.models.Game")
cc.exports.PushCenter = require("src.app.PushCenter");function PlayScene:ctor()self.game_ = Game.new(self)local bg = display.newSprite("img/bg.jpg");bg:setPosition(display.cx, display.cy)self:addChild(bg);local star = display.newSprite("img/star.png")star:setPosition(display.left + 50, display.top - 50)self:addChild(star)--self.starsLabel_ = nil;self.starsLabel_  = cc.Label:createWithSystemFont(self.game_:getStars(), "Arial", 40):move(display.left + 90, display.top - 50):addTo(self)self:onNodeEvent("enter", self.onEnter);self:onNodeEvent("exit", self.onExit);
endfunction PlayScene:onTouchBegan(pTouch,event)print("self  game_:", self, self.game_);print(" pTouch  event:", pTouch:getLocation(), event.name);self.game_:onTouch(event)
endlocal  function addListenerEvent(self)
local this = self;local listener1 = cc.EventListenerTouchOneByOne:create()  --创建一个单点事件监听listener1:setSwallowTouches(true)  --是否向下传递--注册三个回调监听方法listener1:registerScriptHandler(handler(self, self.onTouchBegan),cc.Handler.EVENT_TOUCH_BEGAN )local eventDispatcher = self:getEventDispatcher() --事件派发器eventDispatcher:addEventListenerWithSceneGraphPriority(listener1, self) --分发监听事endfunction PlayScene:onEnter()local this = self;PushCenter:addListener("BUG_ENTER_HOLE_EVENT", function(event)self.starsLabel_:setString(self.game_:getStars())end, self)PushCenter:addListener("PLAYER_DEAD_EVENT", handler(self, self.onPlayerDead_), self)self:getScheduler():scheduleScriptFunc( function(dt)this.game_:step()end, 0.02, false);self:getScheduler():scheduleScriptFunc( function(dt)this.game_:addBug()print("===2");end, 2, false);--[[ self:setTouchEnabled(true)self:addNodeEventListener(cc.NODE_TOUCH_EVENT, function(event)return this.game_:onTouch(event)end)]]addListenerEvent(self);
endfunction PlayScene:onExit()
endfunction PlayScene:onPlayerDead_()
print("--------------------------------game over");local text = string.format("你戳死了 %d 只虫子", self.game_:getDeadCount())cc.ui.UILabel.new({text = text, size = 96}):align(display.CENTER, display.cx, display.cy):addTo(self)cc.ui.UIPushButton.new("img/exit_button.png"):onButtonClicked(function()app:enterScene("MainScene", nil, "Random", 1.0)end):pos(display.cx, display.cy - 200):addTo(self)
endreturn PlayScene

Game.lua


local Game = class("Game")Game.HOLE_POSITION = cc.p(display.cx - 30, display.cy - 75)
Game.INIT_STARS = 50
Game.BUG_ENTER_HOLE_EVENT = "BUG_ENTER_HOLE_EVENT"
Game.PLAYER_DEAD_EVENT = "PLAYER_DEAD_EVENT"local BugAnt = import(".BugAnt")
local BugSpider = import(".BugSpider")
--local PushCenter = require("src.app.PushCenter");function Game:ctor(stage)self.stage_ = stageself.stars_ = Game.INIT_STARSself.bugs_ = {}self.bugsSprite_ = {}self.deadCount_ = 0--    cc(self):addComponent("components.behavior.EventProtocol"):exportMethods()
endfunction Game:getStars()return self.stars_
endfunction Game:getDeadCount()return self.deadCount_
endfunction Game:addBug()local bugif math.random(1, 2) % 2 == 0 thenbug = BugAnt.new()elsebug = BugSpider.new()endbug:setInitPosition(Game.HOLE_POSITION)self.bugs_[#self.bugs_ + 1] = buglocal sprite = display.newSprite(bug:getSpriteName());sprite:setPosition(bug:getPosition())sprite:setRotation(bug:getRotation())self.stage_:addChild(sprite) self.bugsSprite_[#self.bugsSprite_ + 1] = sprite
endfunction Game:bugEnterHole_(index)local bug = self.bugs_[index]local sprite = self.bugsSprite_[index]local action = cc.Sequence:create( cc.FadeIn:create(0.5),cc.CallFunc:create(function()print("============");sprite:removeFromParent(true);end ) )                                      sprite:runAction(action);table.remove(self.bugs_, index)table.remove(self.bugsSprite_, index)self.stars_ = self.stars_ - 1--  self:dispatchEvent({name = Game.BUG_ENTER_HOLE_EVENT})PushCenter.pushEvent("BUG_ENTER_HOLE_EVENT");audio.playSound("sfx/bug_enter.wav")if self.stars_ <= 0 thenself:playerDead_()end
endfunction Game:playerDead_()-- self:dispatchEvent({name = Game.PLAYER_DEAD_EVENT})PushCenter.pushEvent("PLAYER_DEAD_EVENT");audio.playSound("sfx/player_dead.wav")
endfunction Game:step()if self.stars_ <= 0 then return endfor i = #self.bugs_, 1, -1 dolocal bug = self.bugs_[i]bug:step()local sprite = self.bugsSprite_[i]sprite:setPosition(bug:getPosition())if bug:getDist() <= 0 thenself:bugEnterHole_(i)endend
endfunction Game:bugDead_(index)local bug = self.bugs_[index]local sprite = display.newSprite(bug:getDeadSpriteName());sprite:setPosition(bug:getPosition())sprite:rotation(math.random(0, 360))self.stage_:addTo(sprite)local action = c.Sequence:create( cc.FadeIn(0.5),cc.DelayTime:create(2.0),cc.CallFunc:create(function( )self.bugsSprite_[index]:removeSelf()table.remove(self.bugs_, index)table.remove(self.bugsSprite_, index)self.deadCount_ = self.deadCount_ + 1audio.playSound("sfx/bug_dead.wav")end ) )                                      sprite:runAction(action);endfunction Game:onTouch(event)if self.stars_ <= 0 then return endif event.name == "began" thenlocal x, y = event.x, event.yfor i = #self.bugs_, 1, -1 dolocal bug = self.bugs_[i]if bug:checkTouch(x, y) thenself:bugDead_(i)endendendreturn false
endreturn Game

 

这篇关于Cocos2d+Lua 游戏开发中的技巧系列之一 Lua require的语法糖的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1135768

相关文章

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

这15个Vue指令,让你的项目开发爽到爆

1. V-Hotkey 仓库地址: github.com/Dafrok/v-ho… Demo: 戳这里 https://dafrok.github.io/v-hotkey 安装: npm install --save v-hotkey 这个指令可以给组件绑定一个或多个快捷键。你想要通过按下 Escape 键后隐藏某个组件,按住 Control 和回车键再显示它吗?小菜一碟: <template

Hadoop企业开发案例调优场景

需求 (1)需求:从1G数据中,统计每个单词出现次数。服务器3台,每台配置4G内存,4核CPU,4线程。 (2)需求分析: 1G / 128m = 8个MapTask;1个ReduceTask;1个mrAppMaster 平均每个节点运行10个 / 3台 ≈ 3个任务(4    3    3) HDFS参数调优 (1)修改:hadoop-env.sh export HDFS_NAMENOD

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

OpenHarmony鸿蒙开发( Beta5.0)无感配网详解

1、简介 无感配网是指在设备联网过程中无需输入热点相关账号信息,即可快速实现设备配网,是一种兼顾高效性、可靠性和安全性的配网方式。 2、配网原理 2.1 通信原理 手机和智能设备之间的信息传递,利用特有的NAN协议实现。利用手机和智能设备之间的WiFi 感知订阅、发布能力,实现了数字管家应用和设备之间的发现。在完成设备间的认证和响应后,即可发送相关配网数据。同时还支持与常规Sof

活用c4d官方开发文档查询代码

当你问AI助手比如豆包,如何用python禁止掉xpresso标签时候,它会提示到 这时候要用到两个东西。https://developers.maxon.net/论坛搜索和开发文档 比如这里我就在官方找到正确的id描述 然后我就把参数标签换过来

购买磨轮平衡机时应该注意什么问题和技巧

在购买磨轮平衡机时,您应该注意以下几个关键点: 平衡精度 平衡精度是衡量平衡机性能的核心指标,直接影响到不平衡量的检测与校准的准确性,从而决定磨轮的振动和噪声水平。高精度的平衡机能显著减少振动和噪声,提高磨削加工的精度。 转速范围 宽广的转速范围意味着平衡机能够处理更多种类的磨轮,适应不同的工作条件和规格要求。 振动监测能力 振动监测能力是评估平衡机性能的重要因素。通过传感器实时监

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl

【生成模型系列(初级)】嵌入(Embedding)方程——自然语言处理的数学灵魂【通俗理解】

【通俗理解】嵌入(Embedding)方程——自然语言处理的数学灵魂 关键词提炼 #嵌入方程 #自然语言处理 #词向量 #机器学习 #神经网络 #向量空间模型 #Siri #Google翻译 #AlexNet 第一节:嵌入方程的类比与核心概念【尽可能通俗】 嵌入方程可以被看作是自然语言处理中的“翻译机”,它将文本中的单词或短语转换成计算机能够理解的数学形式,即向量。 正如翻译机将一种语言