【Lua热更新】上篇

Lua 热更新 - 上篇

下篇链接:【Lua热更新】下篇

文章目录

  • `Lua` 热更新 - 上篇
    • 一、`AssetBundle`
      • 1.理论
      • 2. AB包资源加载
    • 二、`Lua` 语法
      • 1. 简单数据类型
      • 2.字符串操作
      • 3.运算符
      • 4.条件分支语句
      • 5.循环语句
      • 6.函数
      • 7. table数组
      • 8.迭代器遍历
      • 9.复杂数据类型 - 表
        • 9.1字典
        • 9.2类
        • 9.3 表的公共操作
      • 多脚本执行
      • 10.协同程序
      • 11.元表
        • 11.1 index
        • 11.2 newindex
        • 11.3其他操作
      • 12.面向对象
        • 12.1封装
        • 12.2继承
        • 12.3多态
      • 12 .【重要】面向对象汇总
      • 13.自带库
      • 13.垃圾回收
    • 三、`XLua`热更新

一、AssetBundle

1.理论

  • AB包是什么:特定于平台的资产压缩包,有点类似压缩文件,资产包括一些模型、贴图、预设体、音效、材质球等等

  • AB包作用:Resoureces打包时只读无法修改,AB包存储位置可以自定义,压缩方式自定义,后期可以动态更新。AB包可以减少包体大小,压缩资源减少初始包大小。可以通过热更新更新脚本或者资源

  • 使用

    在这里插入图片描述

2. AB包资源加载

  • 同步加载
// 加载AB包
AssetBundle ab = AssetBundle.LoadFromFile(Application.streamingAssetsPath + "/model");
// 加载AB包中资源
// 通过泛型加载
GameObject cube = ab.LoadAsset<GameObject>("cube");
// 通过type加载
GameObject sphere = ab.LoadAsset("sphere", typeof(GameObject)) as GameObject;
// AB包不能加载两次会报错
// AssetBundle ab2 = AssetBundle.LoadFromFile(Application.streamingAssetsPath + "/model");
  • 异步加载
public Image img;
IEnumerator LoadABRes(string abName, string resNAme)
{// 异步加载ab包AssetBundleCreateRequest abcr = AssetBundle.LoadFromFileAsync(Application.streamingAssetsPath + "/" + abName + "." + resNAme);yield return abcr;AssetBundleRequest abq = abcr.assetBundle.LoadAssetAsync(resNAme, typeof(Sprite));yield return abq;img.sprite = abq.asset as Sprite;
}
  • AB包卸载
AssetBundle ab2 = AssetBundle.LoadFromFile(Application.streamingAssetsPath + "/model");
// 卸载所有AB包
// true会把场景上加载的AB包资源也卸载了
AssetBundle.UnloadAllAssetBundles(false);
// 卸载单独的AB包
ab2.Unload(false);

二、Lua 语法

1. 简单数据类型

  • lua所有变量声明不需要申明变量类型
  • lua可以改变数据类型,自动识别
  • 通过type函数获得类型
  • lua中使用未声明的变量默认是nil
-- 【nil类型】:类似 C#中的null
a = nil
print(a)
print(type(a))-- 【number类型】:表示所有数值类型
a = 1
print(a)
print(type(a))
a = 10.2
print(a)
print(type(a))-- 【string类型】:字符串类型,lua中单引号或双引号都是字符串与C#中不同,lua中没有char
a = "123"
print(a)
a = '111'
print(a)
print(type(a))-- 【boolean类型】
a = true
print(a)
print(type(a))

2.字符串操作

  • lua中文占三个字符
-- 获得字符串长度
str = "hellowrold"
print(#str)
str = "你好"
print(#str)-- 字符串拼接
print("你好".."世界")
print(string.format("我%d岁了", 18))-- 转字符串
str = tostring(true)
print(str)-- 字符串转ASCII码
a = string.byte("Lua", 1)
print(a)
-- ASCII码转字符
str = string.char(a)
print(str)

3.运算符

  • 字符串可以自动转成number
  • 整数相除可以转换为浮点数
  • 不支持自增自减++ –
  • 不支持位运算、三目运算符
  • Luanilfalse 才是假
  • lua中逻辑运算符支持短路
-- 算数运算符
print("3.14" + 2) -- 输出5.14
print(3 / 2) -- 输出1.5
print(2 ^ 2) -- 支持幂运算-- 条件运算符
print(100 ~= 5) -- 不等于符号,与C#不同(!= )-- 逻辑运算符
-- lua中逻辑运算符支持短路
print(true and false) -- &&与
print(true or false) -- ||或
print(not true) -- !非
-- lua中and or可以链接任意类型
print(1 and 2) -- 返回 2
print(nil and 1) -- 返回 nil
-- 使用逻辑运算模拟三目运算符
res = (x > y) and x or y

4.条件分支语句

  • 不支持switch
a = 6
-- if 语句
if a > 5 thenprint("success!")
end-- if-else语句
if a > 5 thenprint("succcess!")
elseprint("fault!")
end-- if-else if语句
if a == 5 thenprint("是5!")
elseif a == 6 thenprint("是6!")
end

5.循环语句

-- while
num = 0
while num < 5 donum = num + 1
end
print(num)-- do while
num = 6
repeatnum = num + 1
until num > 5
print(num)-- for
-- for 变量, 结束值,增量值(省略默认为1)
for i = 1, 5, 2 doprint(i)
end

6.函数

-- 1.1声明
function myFirstFun()print("我的第一个函数")
end
-- 调用
myFirstFun()-- 2.1有参函数
function testFun1(name)print("我的名字是"..name)
end
testFun1("小猫")-- 3.1有返回值函数
-- 3.1.1一个返回值
function testFun2()return "yes"
end
temp = testFun2()
print(temp)
-- 3.1.2多个返回值
function testFun3()return "paramete1","paramete2"
end
temp1, temp2 = testFun3()
print(temp1..temp2)-- 4.1类型
-- 不用写函数名
func1 = function()print("这是一个函数")
end
print(type(func1))
func1()-- 5.1变长参数
function testFun4(...)arg = {...}for i = 1, #arg doprint(arg[i])end
end
testFun4(1, 2, 3)-- 6.1函数嵌套
-- 闭包情况
function myFunC(x)return function(y)return x + yend
endmyfunc = myFunC(1)
ans = myfunc(2)
print("ans = "..ans)

7. table数组

  • lua中下标从1开始,越界读取最后一个
  • 表中存在nil会打断表长
-- 1.1 数组声明
a = {1, 2, true, "name", nil}
print(a[1]) -- 下标从1开始
print(#a) -- 输出:4-- 1.2 多维数组
a = {{1, 2, 3}, {4, 5, 6, 7}}
print(a[1][1])
for i = 1, #a dob = a[i]for j = 1, #b doprint(b[j])end
end

8.迭代器遍历

  • pairs遍历所有都能获得建议使用
a = {[0] = 1, 2, [-1] = 3, 4, 5}-- 1.1 ipairs遍历
-- 只能找从1开始下标,小于等于0得不到
-- 只能找连续索引的,如果中断后面无法获得
for k, v in ipairs(a) doprint("key = "..k.." value = ".. v)
end-- 1.2 pairs遍历
-- 所有都能获得
for k, v in pairs(a) doprint("key = "..k.." value = ".. v)
end
-- 只遍历键
for k in pairs(a) doprint("key = "..k)
end

9.复杂数据类型 - 表

9.1字典
fruitDic = {["apple"] = 10, ["banana"] = 5}
print(fruitDic["apple"])
print(fruitDic.banana) -- 只有字符串可以通过.获得-- 遍历
for k, v in pairs(fruitDic) doprint(k,v) -- 传入多个参数print
end
9.2类
  • lua中不存在面向对象
student = {name = "lily",age = 10,sex = true,changeSex = function(v)student.sex = vend,printSex = function()print(student.sex)end,printName = function(m)print(m.name)end
}student.printSex()
student.changeSex(false)
student.printSex()-- 冒号调用
-- 冒号调用默认第一个参数是调用者
-- .调用不能
student:printName()-- 外部添加函数
-- 方法一:
function student.myScore(v)print(v + 100)
end-- slef表示默认传入的第一个参数
-- 方法二:
function student:printAge()print(self.age)
end
-- 冒号声明冒号调用
student:printAge()
9.3 表的公共操作
t1 = {{name = "小猫", age = 1}, {name = "小狗", age = 2}}
t2 = {name = "小鸟", age = 2}-- 1.1插入
-- t2 插入到 t1 中
table.insert(t1, t2)
print(#t1)-- 1.2删除
-- 删除 t1 的最后一个元素
table.remove(t1)
print(t1[2].name)
-- 删除指定下标
table.remove(t1, 1)
print(t1[1].name)-- 1.3排序
t = {5, 6, 1, 3, 9, 2}-- 默认升序
table.sort(t)
print("----降序-----")
for _,v in pairs(t) doprint(v)
end-- 降序
table.sort( t, function(a, b)return a > b
end )
print("----降序-----")
for _,v in pairs(t) doprint(v)
end-- 1.4 拼接
tb = {"ccc", "aaa", "bbb"}
str = table.concat( tb, ";")
print(str)

多脚本执行

-- 1.1 局部变量
function myFunC()local x = 1y = 1
endmyFunC()
print(x) -- x是局部变量
print(y) -- y是全局变量-- 1.2 多脚本执行
-- require("脚本名") 执行其他脚本,只能加载一次
require("testLua")
-- 获得其他脚本的全局变量,局部变量获取不到
print(myTest)-- 1.3脚本卸载
-- 脚本是否被加载过
isLoad = package.loaded["testLua"]
print(isLoad)
-- 卸载脚本
package.loaded["testLua"] = nil-- 1.4 大G表
-- 大G表是一个总的table表,存储了所有全局变量在里面
for k,v in pairs(_G) doprint(k, v)
end

在这里插入图片描述

require 脚本可以 return 一个数据返回出去得到

10.协同程序

fun = function()print("使用协程")local i = 1while true doprint(i)i = i + 1coroutine.yield(i)end
end-- 返回一个线程
co = coroutine.create(fun)
print("类型:"..type(co))
coroutine.resume(co) -- 输出1
isOK, temp1 = coroutine.resume(co) -- 输出2
print(isOK,temp1) -- 第一个参数返回协程状态,第二个参数返回协程返回值-- 返回的是函数
co2 = coroutine.wrap(fun)
print("类型:"..type(co2))
co2() -- 输出1
temp1 = co2() -- 输出2
print(temp1) -- 直接返回协程返回值-- 获得协程状态
-- 传入协程对象,返回状态:dead结束 susdeaded暂停 running进行
back = coroutine.status(co)
print(back)

11.元表

11.1 index
  • index取值时,使用当子表中找不到某属性时会到元素中__index指定的表去查找索引
  • index支持向上查找原表的index
meta = {name = "father",age = 10,
}
sonTable = {name = "son"
}
setmetatable(sonTable, meta)
print(sonTable.age) -- 输出 nil-- 设置__index
meta.__index = meta
print(sonTable.age) -- 输出 10
11.2 newindex
  • newindex当赋值时,如果赋值不存在索引,将会赋值到newindex所指向的表中
meta = {name = "father"
}
sonTable = {
}
setmetatable(sonTable, meta)sonTable.name = "son"
print(sonTable.name) -- 输出sonmeta.__newindex = {}
print(sonTable.age) -- 输出nil
11.3其他操作
meta = {name = "father",__tostring = function(t)return t.nameend,__call = function(t)print(t)print("call!")end,__add = function(t1, t2)return t1.name..t2.nameend
}
sonTable = {name = "son"
}-- 设置元表:参数一 子表 参数二 元表
setmetatable(sonTable, meta)-- 1.1 __tostring
-- 当子表被当作字符串使用时会调用tostring方法
-- 默认传递第一个参数为调用者自己
print(sonTable)-- 输出son-- 1.2 __call
-- 子表被当作函数使用时会调用call方法
-- 默认传递第一个参数为调用者自己
sonTable() -- 输出 son call!-- 1.3运算符重载
-- __add 加法
print(sonTable + meta) -- 输出sonfather
-- __sub 减法、__mul 乘法、 __div 除法、 __mod 取余、 __pow 幂、 
-- __eq  相等== 原表一致返回true、__lt  小于、 __le  小于等于、__concat 拼接

12.面向对象

12.1封装
-- 1.1 封装
Object = {}
Object.id = 1-- 冒号调用,默认使用调用者作为第一个参数
function Object : new()-- self,获得默认传入的第一个参数-- 返回一个新对象(表),使用局部变量-- 设置对象的元表,和__indexlocal obj = {}self.__index = selfsetmetatable(obj, self)return obj
endlocal my = Object : new()
print(my.id) -- 输出1,因为使用了__indexmy.id = 99
print(my.id) -- 输出99,没有使用__newindex不会改变元表
print(Object.id) -- 输出1-- 封装一个函数
function Object : TestFun()print(self.id)
endmy:TestFun() -- 输出99
Object:TestFun() -- 输出1
12.2继承
-- 为父类写一个继承方法
function Object : subClass(className)-- _G表,是一个总表存储所有声明了的全局变量,以键值对形式存储-- 在_G中创建一个表_G[className] = {}-- 用元表完成继承local obj = _G[className]self.__index = selfsetmetatable(obj, self)
end-- Person继承Object
Object:subClass("Person")
-- 使用Object中的new对象方法
local p1 = Person:new()
print(p1.id)
12.3多态

在这里插入图片描述

-- Entity 继承 Object
Object : subClass("Entity")
-- 为 Object 添加变量和方法
Object.Speed = 0
function Object : GetSpeed()print("这是Object的方法")self.Speed = self.Speed + 1print(self.Speed)return self.Speed
end-- Player 继承 Entity
Entity : subClass("Player")-- 创建Player 对象 p1
local p1 = Player:new()
p1:GetSpeed() -- 输出:这是Object的方法-- 子类Player重写方法
function Player : GetSpeed()print("这是Player重写的方法")return self.Speed
end
p1:GetSpeed() -- 输出:这是Player重写的方法-- 构建base保留父类方法调用 : 如图-- 创建Player 对象 p2
local p2 = Player:new()
p2:GetSpeed() -- 输出:这是Player重写的方法-- 【注意】有误的
p1.base:GetSpeed() -- 输出:Speed = 1
p2.base:GetSpeed() -- 输出:Speed = 2
-- 【注意】正确的,使用点调用否则使用的是父类的变量
function Player : GetSpeed()self.base.GetSpeed(self)
end
p1:GetSpeed()
p2:GetSpeed()
-- 或者
p1.base.GetSpeed(p1)
p2.base.GetSpeed(p2)

12 .【重要】面向对象汇总

-- 基类
Object = {}
-- 实例化对象方法
function Object : new()local obj = {}self.__index = selfsetmetatable(obj, self)return obj
end
-- 继承方法
-- className:类名(字符串)
function Object : subClass(className)_G[className] = {}local obj = _G[className]self.__index = selfobj.base = selfsetmetatable(obj, self)
end

13.自带库

-- 时间
-- 系统时间
print(os.time())
print(os.time({year = 2024, month = 12, day = 17}))
local nowTime = os.date("*t")
for k, v in pairs(nowTime) doprint(k, v)
end-- 数学运算
print(math.abs(-11)) -- 绝对值
print(math.deg(math.pi)) -- 弧度转角度
print(math.cos(math.pi)) -- 三角函数
print(math.floor(2.6)) -- 向下取整
print(math.ceil(2.6)) -- 向上取整
print(math.max(1, 2))
print(math.min(1, 2))
print(math.modf(1.2)) -- 返回整数和小数部分
print(math.pow(2, 2)) -- 幂运算
math.randomseed(os.time()) -- 更换随机数种子
print(math.random(100)) -- 随机数
print(math.random(100)) -- 随机数
print(math.sqrt(2)) -- 开方-- 路径
print(package.path)

13.垃圾回收

三、XLua热更新

下篇链接:【Lua热更新】下篇

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/493453.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

React图标库: 使用React Icons实现定制化图标效果

React图标库: 使用React Icons实现定制化图标效果 图标库介绍 是一个专门为React应用设计的图标库&#xff0c;它包含了丰富的图标集合&#xff0c;覆盖了常用的图标类型&#xff0c;如FontAwesome、Material Design等。React Icons可以让开发者在React应用中轻松地添加、定制各…

如何使用 WebAssembly 扩展后端应用

1. WebAssembly 简介 随着互联网的发展&#xff0c;越来越多的应用借助 Javascript 转到了 Web 端&#xff0c;但人们也发现&#xff0c;随着移动互联网的兴起&#xff0c;需要把大量的应用迁移到手机端&#xff0c;随着手端的应用逻辑越来越复杂&#xff0c;Javascript 的解析…

Fastdfs V6.12.1集群部署(arm/x86均可用)

文章目录 一、 Fastdfs 介绍二、部署 信息三、步骤tracker/storage 机器的 compose 内容storage 机器的 composetracker 与 storage 启动目录层级与配置文件测试测试集群扩容与缩减注意事项 一、 Fastdfs 介绍 FastDFS 是一款高性能的分布式文件系统&#xff0c;特别适合用于存…

maven-resources-production:ratel-fast: java.lang.IndexOutOfBoundsException

Maven生产环境中遇到java.lang.IndexOutOfBoundsException的问题&#xff0c;尝试了重启电脑、重启IDEA等常规方法无效&#xff0c;最终通过直接重建工程解决了问题。 Rebuild Project 再启动OK

1. JasperSoft介绍与安装

Jaspersoft介绍 Jaspersoft是一款开源的&#xff0c;强大灵活并且使用广泛的报表软件。能够展示丰富的页面内容&#xff0c;并将之转换成PDF、HTML或者XML格式&#xff0c;该库完全由Java写出&#xff0c;可以用于在各种Java应用程序&#xff0c;非常适合Java开发者用来做报表生…

知网研学 | 知网文献(CAJ+PDF)批量下载

知网文献&#xff08;CAJPDF&#xff09;批量下载 一、知网研学安装二、插件及脚本安装三、CAJ批量下载四、脚本下载及PDF批量下载浏览器取消拦截窗口 一、知网研学安装 批量下载知网文件&#xff0c;格式为es6文件&#xff0c;需使用知网研学软件打开&#xff0c;故需先安装该…

WeakAuras NES Script(lua)

WeakAuras NES Script 修星脚本字符串 脚本1&#xff1a;NES !WA:2!TMZFWXX1zDxVAs4siiRKiBN4eV(sTRKZ5Z6opYbhQQSoPtsxr(K8ENSJtS50(J3D7wV3UBF7E6hgmKOXdjKsgAvZFaPTtte0mD60XdCmmecDMKruyykDcplAZiGPfWtSsag6myGuOuq89EVDV9wPvKeGBM7U99EFVVVV33VFFB8Z2TJ8azYMlZj7Ur3QDR(…

[数据结构] 链表

目录 1.链表的基本概念 2.链表的实现 -- 节点的构造和链接 节点如何构造? 如何将链表关联起来? 3.链表的方法(功能) 1).display() -- 链表的遍历 2).size() -- 求链表的长度 3).addFirst(int val) -- 头插法 4).addLast(int val) -- 尾插法 5).addIndex -- 在任意位置…

springmvc的拦截器,全局异常处理和文件上传

拦截器: 拦截不符合规则的&#xff0c;放行符合规则的。 等价于过滤器。 拦截器只拦截controller层API接口。 如何定义拦截器。 定义一个类并实现拦截器接口 public class MyInterceptor implements HandlerInterceptor {public boolean preHandle(HttpServletRequest reque…

RestTemplate远程调用、服务注册、

一.RestTemplate Spring给我们提供了一个RestTemplate的API&#xff0c;可以方便的实现Http请求的发送。 同步客户端执行HTTP请求&#xff0c;在底层HTTP客户端库(如JDK HttpURLConnection、Apache HttpComponents等)上公开一个简单的模板方法API。RestTemplate通过HTTP方法为常…

台球助教平台系统开发APP和小程序信息收藏功能需求解析(第十二章)

以下是开发台球助教系统客户端&#xff08;APP&#xff0c;小程序&#xff0c;H5&#xff09;几端的信息收藏功能的详细需求和功能说明&#xff0c;内容比较详细&#xff0c;可以说是一个教科书式的详细说明了&#xff0c;这套需求说明不仅仅用在我们的台球助教系统程序上&…

React系列(八)——React进阶知识点拓展

前言 在之前的学习中&#xff0c;我们已经知道了React组件的定义和使用&#xff0c;路由配置&#xff0c;组件通信等其他方法的React知识点&#xff0c;那么本篇文章将针对React的一些进阶知识点以及React16.8之后的一些新特性进行讲解。希望对各位有所帮助。 一、setState &am…

【开源免费】基于SpringBoot+Vue.JS房屋租赁管理系统(JAVA毕业设计)

本文项目编号 T 091 &#xff0c;文末自助获取源码 \color{red}{T091&#xff0c;文末自助获取源码} T091&#xff0c;文末自助获取源码 目录 一、系统介绍二、数据库设计三、配套教程3.1 启动教程3.2 讲解视频3.3 二次开发教程 四、功能截图五、文案资料5.1 选题背景5.2 国内…

netcore 集成Prometheus

一、安装包 <ItemGroup><PackageReference Include"prometheus-net" Version"8.2.1" /><PackageReference Include"prometheus-net.AspNetCore" Version"8.2.1" /> </ItemGroup> 二、添加代码 #region Pro…

CLION中运行远程的GUI程序

在CLION中运行远程GUI程序&#xff0c;很有可能会遇到下面错误 Gtk-WARNING **: cannot open display: 这是因为远程的GUI程序不能再本地机器上显示。这个问题一般有两种解决方法 通过SSH的ForwardX11的方法&#xff0c;就是将远程的GUI程序显示到本地机器上&#xff0c;一般在…

datasets 笔记:加载数据集(基本操作)

参考了huggingface的教程 1 了解数据集基本信息&#xff08; load_dataset_builder&#xff09; 在下载数据集之前&#xff0c;通常先快速了解数据集的基本信息会很有帮助。数据集的信息存储在 DatasetInfo 中&#xff0c;可能包括数据集描述、特征和数据集大小等信息。&…

Java期末复习暨学校第十三次上机课作业

Java期末复习暨学校第十三次上机课作业&#xff1a; &#xff08;1&#xff09;&#xff1a;掌握正则表达式的使用 第一题&#xff1a; 第13行代码为正则表达式&#xff0c;中国内地的手机号必须是11个数字&#xff1a; &#xff08;1&#xff09; ^1&#xff1a;是该电话号码…

aosp15 - Activity生命周期切换

本文探查的是&#xff0c;从App冷启动后到MainActivity生命周期切换的系统实现。 调试步骤 在com.android.server.wm.RootWindowContainer#attachApplication 方法下断点&#xff0c;为了attach目标进程在com.android.server.wm.ActivityTaskSupervisor#realStartActivityLock…

【libuv】Fargo信令1:client发connect消息给到server

tcp 单机测试,进行模拟 (借助copilot实现) 【Fargo】28:字节序列client发connect消息给到serverserver 收到后回复ack给到客户端程序借助copilot实现。项目构建 Console依赖于Halo.dll提供的api,Halo 依赖于 Immanuel, 运行效果 遗留问题 客户端似乎么有逻辑收到ack做处理各…

libmodbus安装使用

要配置和编译 libmodbus&#xff0c;您需要确保安装了所有必要的依赖项&#xff0c;并按照正确的步骤进行操作。以下是详细的环境配置和编译指南&#xff0c;适用于不同的操作系统。 1. Linux (Debian/Ubuntu) 安装依赖项 首先&#xff0c;确保您的包列表是最新的&#xff1…