Lua热更新(Lua)

时间:2024-04-01 11:04:49

--  [[]] print

下载Lua For Windows   Sublime Text(仅用于演示,实际项目使用VsCode)       Ctrl+B运行

语法基础

基础类型:nil number string boolean

运算符:and-or-not  ~= ^  if-then-end-elseif-else  while-do-end  repeat-until   for i=1,10 do end

 函数:function  

table: pairs ipairs {["键"]=值,}   {类,可以有函数}  table.insert remove sort  concat

 local require package.loaded[]   _G return

协程:coroutine.creat-wrap    coroutine.resume()-()   coroutine.yield()  coroutine.status-running

元表:setmetatable() __tostring=  __call=  __add等  __index __newIndex

面向对象:封装 继承 多态 base传入self

os.time  os.date("*t") math.cos deg rad max min floor cliet modf

collectgarbage("count")   collectgarbage("collect")

变量

  type可以得到类型,而它本身返回一个string

nil string number boolean

字符串

如果是括号,不用加双引号

即使s2是其他类型也可以拼接,默认拼接成字符串

string提供的公共方法:        注意只是返回一个新字符串,不会改变原字符串对应输出

第二个返回值是,修改了几处

运算符

短路就是前面为false后面就不会执行

三目运算符也不支持

条件分支/循环语句

可以组合

dowhile比较特殊,在c#里面是进入条件,而这里是结束条件。 且关键词也不同

默认递增,且默认增1

递减

函数

lua的语法全部是从上到下调用的。在下面定义的函数不能在上面调用,c#可以,但lua不行

用的是变量则调用变量

参数可以不写类型

参数可以随便传。  返回值也不用指定类型,在外面接收即可。 数量也无限制,类似元组

打出function,按tab可以补全

不支持重载

function默认的三个点其实就是变长参数。 用一个表把他接收起来(类型无限制),然后按长度输出

用一个变量接收再返回也可以,但其实function本身就是一个变量,直接返回也是没问题的

会输出15   充当返回值的函数不要写函数名,它是变量

table

所有复杂类型都是talbe

例如这里#a,会输出2,后面断了

如果中间为空会断掉,可能遍历不完

  

可以这样写,但一般不建议用负数

顺带一提,#aa=3,aa[1] [2] [3] 分别是2,3,5  并没有被分开

加两道题:打印5打印2

空一个不会断,空两个会断掉。 一般不建议你自定义索引

迭代器遍历

中间没有4,断开了,所以5也得不到,这是ipairs

 pairs则全部打印出来了

这种写法可以只遍历键,只写一个变量

字典

 和之前写的键值对类似为i确保万无一失还是用中括号填键访问比较稳妥

置空应该可以认为是删除了。 或者直接删代码(有风险)

paint(k,v,1,2,3)其实打印多少个参数都可以,print会自动空行。

   也可以只得到键,方法同上。但不能只得到值,这种写法本质上还是得到了键与值。

←只不过他没有使用键,只打印值就好了

类和结构体

一切都是模仿

方法类似,点出来调用建议都用冒号申明

而且lua中没有this.age这个东西

申明函数时如果不用点,而用冒号的话,就会多出一个self关键字,必须配合冒号调用

总结:要么指明是谁的,要么传参时传入自己。 冒号调用等于默认传入自己,如果像↑这种我根本没有写参数,那么self就默认是传入的第一个参数。

可以是 Student:Speak2    Student.Speak2(Student)    两种都可以 self必须配合冒号调用

如果搞不清,可以简单一点调用→不是非要用冒号和self

但要搞清楚他俩的区别

插入移除排序拼接

↑这样申明是可以的,表里面可以套表

t2插入t1后面

如果传参则按参数来,不传参则默认移除最后一个

直接调用默认升序true就是a放前面

一般只用于字符串

多脚本执行

我们之前声明的都是全局变量,哪怕是在函数内部申明的,也是全局变量

require(字符串形式)

 注意会从头到尾执行一遍 test文件如这里全局变量testA就可以正常打印,local则不行

可以执行另一个lua文件(这里的test是文件名,必须加引号) 同时允许获取其中的全局变量

其他脚本只能加载一次。想要再次执行,只能卸载过后再次加载     注意是中括号

本地变量例外,不会被存到大G表中。  G表也可以用来存东西

但仍然可以通过返回一个本地变量,来使用本地变量只需要在外部去接收它使用

多变量三目运算符

另外函数可以同时返回多个返回值  非-not

分别输出2,1,nil    可以利用此特性模拟三目运算符

协程

输出这几种方法都可以,注意f1是一个函数

对应wrap

注意调用一次协程只执行一次,lua特性是从上到下只执行一次想要多次执行协程就要多次调用

两种方法都有返回值,第一种会额外返回执行状态(所以可以用两个变量接收),第二种只会返回值

实测,只能得到create的协程的状态

running状态和 coroutine.running() 只能写在协程内部去得,写在外面是得不到“正在运行”的状态的

元表

输出表默认会打印类型和编码,但如果重写了输出子表会调用元表的tostring方法如果填参数,则参数a代表子表自己

参数a代表子表自己

第二个参数开始,才代表你传进去的参数。  一个表本身是不支持像函数一样调用的。必须指定了元表,并设置了__call 才可以像函数一样调用。

+ - *  / % ^  == < <= ..同理,注意没有> 和>+,只有小于指定同一张元表才会得true

也可以在meta.__index={} 申明一个空表        可以寻找元表的元表,但前提给爷表的index也指定好

这个倒是没有index重要这个不会管元表

rawget rawset getmetatable  后面几个用的不多

Lua面向对象

封装就是new一个对象,接收,访问该表是访问它的元表,但增加却是增加的它自己

继承,是父类自己提供的方法,通过该方法创建一个子类,自动指定父类为元表,也可以new

封装

先申明一个空表,然后增加方法,该方法又返回一个空表,只不过又用到了元表的知识得到id

这时候外面myobj就等于里面local obj了,只有建立元表关系,并指定index才能访问其中的id

输出1,2为什么?原因:myobj第一次调用test,把自己作为参数传递,但它其中没有id',所以使用了元表当中的id。但之后又设置了id,就输出自己的

继承

继承用大G表,因为它是全局的。封装不用,因为它是本地的

object表中加了一个方法,传一个字符串可以在大G表中申明一个空表。

注意区分传入时是字符串,调用则只需要名字  类似字典的访问  因为都是键值对

写的很清楚了

如果改变了class1的name,不影响父类的值,相当于子类拥有了自己的值

多态

子类父类都有同一个方法,子类重写了该方法,执行不同的逻辑。 c#中通过base. 可以保留父类方法,lua中则需要一些操作

在上一步继承方法里,给子类定义一个base属性,让它就等于父类

注意冒号调用操作的是父类数据,不要这么写

这三次调用实际上都是在操作GameObject这个大类里的数据,并不是他们自己的

总结:

--多态
--申明一个新的类
Object:subClass("GameObject")
--成员变量
GameObject.posX = 0
GameObject.posY = 0
--成员方法
function GameObject:Move()
	self.posX = self.posX + 1
	self.posY = self.posY + 1
end

--实例化对象使用
local obj = GameObject:new()
print(obj.posX)
obj:Move()
print(obj.posX)


--申明一个新的类 Player 继承 GameObject
GameObject:subClass("Player")
--多态 重写了 GameObject的Move方法
function Player:Move()
	--base调用父类方法 用.自己传第一个参数
	self.base.Move(self)
end
print("****")
--实例化Player对象
local p1 = Player:new()
print(p1.posX)
p1:Move()
print(p1.posX)

自带库

os.time得到的是这种 所以我们一般自定规则  注意os.date("*t")得到一张表

遍历输出

也可以点出单个值   ↑

????就是180°

30°是弧度,一π等于180°。 面板上显示的是角度,如cos(30°)=cos(30*mathf.deg2rad)

其实用^ 就行了种子变了随机数才会变一般不会改

垃圾回收

尽量是手动触发GC,不要自动GC。 比如过场景时手动去触发GC