Lua更多语法与使用

文章目录

  • 目的
  • 错误处理
  • 元表和元方法
  • 垃圾回收
  • 协程
  • 模块
  • 面向对象
  • 总结

目的

在前一篇文章: 《Lua入门使用与基础语法》 中介绍了一些基础的内容。这里将继续介绍Lua一些更多的内容。

同样的本文参考自官方手册:
https://www.lua.org/manual/

错误处理

下面代码可以直接测试相关内容:

-- 使用 assert (v [, message]) 可以检查条件v,如果失败则抛出错误信息message--[[
使用 error (message [, level]) 抛出错误信息message
level为1时(默认值)会在message头部添加调用error函数的位置信息;
level为2时会添加调用error的函数被调用的位置信息;
level为0时不添加额外信息。
]]-- 使用 pcall() (f [, arg1, ···]) 执行函数并捕获可能的错误。执行无错误返回true,否则返回false和错误对象。function func1(arg)error(arg) -- 抛出错误
endret, msg = pcall(func1, "hello naisu!")
print("pcall: ", ret, msg)-- xpcall (f, msgh [, arg1, ···]) 和pcall类似,第二个参数为错误处理函数function func2()print(debug.traceback())
endret, msg = xpcall(func1, func2, "hello naisu!")
print("xpcall: ", ret, msg)

元表和元方法

Lua中可以给值(变量)添加一个原表,元表中可以是各种元方法。元方法用于改变该对象的默认行为,比如该对象运算时的行为,格式化输出时的行为等。

使用 setmetatable (table, metatable) 方法向 table 添加一个 metatablemetatablenil 时用于移除元表;使用 getmetatable (object) 返回对象的原表。

常见的运算相关元方法如下:

元方法运算元方法运算元方法运算元方法运算
__add+__sub-__mul*__div/
__mod%__pow^__unm-__idiv//
__band&__bor|__bxor~(异或)__bnot~(非)
__shl<<__shr>>__concat__len#
__eq==__lt<__le<=

除了上面一些基础的运算相关的元方法,Lua中还有更多元方法,下面是个基本的测试:

--[[ 下面是__index ]]
print("__index:")t1 = setmetatable({ a=22 },{ __index = function() return 33 end})print(t1.a)
print(t1.b) -- 访问不存在的元素t2 = setmetatable({ a=22 },{ __index = { b=33 }})print(t2.a)
print(t2.b) -- 访问不存在的元素
print(t2.c) -- 访问不存在的元素--[[ 下面是__newindex ]]
print("__newindex:")t3 = setmetatable({ a=22 },{ __newindex = function (table, key, value) print(#table, key, value) end})print(t3.a)
t3.b = 33   -- 设置不存在的元素时会调用__newindex
print(t3.b) -- __newindex是函数时,值并不会设置到table中t4 = setmetatable({ a=22 },{ __newindex = function (table, key, value) rawset(table, key, value * 2) end})print(t4.a)
t4.b = 33   -- 设置不存在的元素时会调用__newindex
print(t4.b) -- rawset方法将新元素添加到table中t5 = {}
t6 = setmetatable({ a=22 },{ __newindex = t5})
t6.b = 33
print(t6.b) -- __newindex是表时,设置不存在的元素,值并不会设置到table中
print(t5.b) -- __newindex是表时,设置不存在的元素,值会设置到元表中--[[ 下面是__call ]]
print("__call:")t7 = setmetatable({ a=22 },{ __call = function (table, value) print(table.a, value) end})t7(33) -- __call方法会在把表当函数用的时候被调用--[[ 下面是__close ]]
print("__close:")t8 = setmetatable({ a=22 },{ __close = function (table) print(table.a)  end})dolocal var<close> = t8
end
-- close属性的变量会在退出其作用域时调用__close方法--[[ 下面是__tostring ]]
print("__tostring:")t9 = setmetatable({ a=22 },{ __tostring = function (table) return "value: "..table.a end})print(tostring(t9)) -- __tostring方法会在使用tostring函数时被调用

垃圾回收

Lua是带垃圾回收功能的,通常不用手动去控制它,如果有特殊需求的话可以使用 collectgarbage ([opt [, arg]]) 方法来手动控制垃圾收集器。比如使用 collectgarbage("collect") 进行一次完整的垃圾回收、使用 collectgarbage("count") 获得Lua占用的内存数据(单位为1024字节)。

另外如果一个表被其他表引用的话,该表不会被回收,可以用过弱表来处理该情况:

t1 = {}setmetatable(t1, {__mode = "k"}) -- 设置表的key为弱类型k1 = {}
t1[k1] = 22
k2 = {}
t1[k2] = 33
k1 = nil    -- 将k1删除,k1引用的那个表只存在于t1,但t1是key弱引用,所以这个表将被GCcollectgarbage() -- 强制执行一次GC
for k, v in pairs(t1) do print(v) end

协程

Lua中的协程和大多数语言中的协程差不多,提供了一些协程创建、启动、挂起等方法,协程需要主动交出控制权。

  • coroutine.create (f)
    传入函数创建协程返回协程句柄;
  • coroutine.resume (co [, val1, ···])
    运行协程,传入协程句柄和参数;成功时返回true和返回值;失败时返回false和错误信息;
  • coroutine.yield (···)
    协程主动暂停,输入的参数是传递给resume的返回值;
  • coroutine.close (co)
    关闭协程
  • coroutine.wrap (f)
    传入函数创建协程返回函数,调用该函数相当于调用resume;
  • coroutine.isyieldable ([co])
  • coroutine.running ()
    返回正在运行的协程加上一个布尔值,当正在运行的协程是主协程时为true。
  • coroutine.status (co)

下面代码是个简单的测试:

function func()print("协同程序 func 开始执行")local value = coroutine.yield("暂停 func 的执行")print("协同程序 func 恢复执行,传入的值为: " .. tostring(value))print("协同程序 func 结束执行")
end-- 创建协同程序
local co = coroutine.create(func)-- 启动协同程序
local status, result = coroutine.resume(co)
print(result) -- 输出: 暂停 func 的执行-- 恢复协同程序的执行,并传入一个值
status, result = coroutine.resume(co, 233)
print(result) -- 输出: 协同程序 func 恢复执行,传入的值为: 233

模块

当需要多个文件组合实现功能时就涉及到模块的概念的,简单理解就是一个文件引用其他文件:
在这里插入图片描述

面向对象

Lua也支持面向对象的方式,本质其实就是table中同时放变量和函数等。稍稍比table使用多一点的内容是可以使用 : 来定义个使用table中的函数,这样这个函数内部默认会有一个指向自身的 self 对象。

下面代码可以直接测试相关内容:

Class = {}function Class:new(name) -- 注意这里内部的操作,这个方法相当于构造函数obj = {}setmetatable(obj,self)self.__index = selfself.name = namereturn obj
endfunction Class:printName()print(self.name);
endobj1 = Class:new("naisu") -- 新建一个对象print(obj1.name) -- 使用 . 访问属性
obj1:printName() -- 使用 : 调用方法obj2 = Class:new("nx233") 
obj2:printName()

总结

上面的一些Lua的语法功能使得Lua更加灵活和完善,不单单作为脚本,进行大型项目开发也可一用。

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

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

相关文章

嵌入式养成计划-48----QT--信息管理系统:百川仓储管理

一百二十二、信息管理系统&#xff1a;百川仓储管理 122.1 UI界面 122.2 思路 客户端&#xff1a; 用户权限有两种类型&#xff0c;一种是用户权限&#xff0c;一种是管理员权限&#xff0c;登录时服务器端会根据数据库查询到的此用户名的权限返回不同的结果&#xff0c;客户…

学习c#的第五天

目录 C# 运算符 算术运算符 关系运算符 逻辑运算符 位运算符 赋值运算符 其他运算符 C# 中的运算符优先级 C# 运算符 算术运算符 下表显示了 C# 支持的所有算术运算符。假设变量 A 的值为 10&#xff0c;变量 B 的值为 20&#xff0c;则&#xff1a; 运算符描述实例…

JVM在线分析-解决问题的工具一(jinfo,jmap,jstack)

1. jinfo (base) PS C:\Users\zishi\Desktop> jinfo Usage:jinfo <option> <pid>(to connect to a running process)where <option> is one of:-flag <name> to print the value of the named VM flag #输出对应名称的参数-flag [|-]<n…

ElasticSearch7.x - HTTP 操作 - 文档操作

创建文档(添加数据) 索引已经创建好了,接下来我们来创建文档,并添加数据。这里的文档可以类比为关系型数 据库中的表数据,添加的数据格式为 JSON 格式 向 ES 服务器发 POST 请求 :http://192.168.254.101:9200/shopping/_doc 请求体内容为: {"title":"小…

【Python 千题 —— 基础篇】账号登录

题目描述 题目描述 简易登录系统。你的账号密码分别是 “student”&#xff0c;“123456”&#xff1b;请使用 if-else 设计一个简易登录系统&#xff0c;输入账号密码。登陆成功输出 “Welcome !”&#xff0c;登录失败输出 “Login failed !” 输入描述 输入账号和密码。…

idea配置tomcat参数,防止nvarchar保存韩文、俄文、日文等乱码

描述下我的场景&#xff1a; 数据库服务器在远程机器上&#xff0c;数据库使用的Oracle&#xff0c;字符集是ZHS16GBK&#xff0c;但保存韩文、俄文、日文等字段A的数据类型是nvarchar(120)&#xff0c;而nvarchar使用的是Unicode 编码&#xff0c;有点乱。。 遇到的问题&…

【机器学习范式】监督学习,无监督学习,强化学习, 半监督学习,自监督学习,迁移学习,对比分析+详解与示例代码

目录 1. 监督学习 (Supervised Learning): 2. 无监督学习 (Unsupervised Learning): 3. 强化学习 (Reinforcement Learning): 4. 半监督学习 (Semi-Supervised Learning): 5. 自监督学习 (Self-Supervised Learning): 6. 迁移学习 (Transfer Learning): 7 机器学习范式应…

AI:67-基于深度学习的脱机手写汉字识别

🚀 本文选自专栏:AI领域专栏 从基础到实践,深入了解算法、案例和最新趋势。无论你是初学者还是经验丰富的数据科学家,通过案例和项目实践,掌握核心概念和实用技能。每篇案例都包含代码实例,详细讲解供大家学习。 📌📌📌在这个漫长的过程,中途遇到了不少问题,但是…

『Linux升级路』基础开发工具——vim篇

&#x1f525;博客主页&#xff1a;小王又困了 &#x1f4da;系列专栏&#xff1a;Linux &#x1f31f;人之为学&#xff0c;不日近则日退 ❤️感谢大家点赞&#x1f44d;收藏⭐评论✍️ 目录 一、vim的基本概念 &#x1f4d2;1.1命令模式 &#x1f4d2;1.2插入模式 &…

dcat admin 各种问题

样式问题 如何根据条件给表格数据栏添加背景色 use Illuminate\Support\Collection;protected function grid(){return Grid::make(new BookArticle(), function (Grid $grid) {... 其他代码// Collection的完整路径&#xff1a;Illuminate\Support\Collection;$grid->row…

Spark的执行计划

Spark 3.0 大版本发布&#xff0c;Spark SQL 的优化占比将近 50%。Spark SQL 取代 Spark Core&#xff0c;成为新一代的引擎内核&#xff0c;所有其他子框架如 Mllib、Streaming 和 Graph&#xff0c;都可以共享 Spark SQL 的性能优化&#xff0c;都能从 Spark 社区对于 Spark …

笔记:AI量化策略开发流程-基于BigQuant平台(一)

从本文开始&#xff0c;按照AI策略开发的完整流程&#xff08;共七步&#xff09;&#xff0c;上手在BigQuant平台上快速构建AI策略。本文首先介绍如何使用证券代码模块指定股票范围和数据起止日期。重要的事情说三遍&#xff1a;模块的输入端口有提示需要连线的上游数据类型&a…

魔搭社区LLM模型部署实践, 以ChatGLM3为例(一)

魔搭社区LLM模型部署实践&#xff0c; 以ChatGLM3为 例 本文以ChatGLM3-6B为例&#xff0c; 主要介绍在魔搭社区如何部署LLM&#xff0c; 主要包括如下内容&#xff1a; ● SwingDeploy - 云端部署&#xff0c; 实现零代码一键部署 ● 多端部署 - MAC个人笔记本&#xff0c;…

快速入门安装及使用git与svn的区别常用命令

一、导言 1、什么是svn&#xff1f; SVN是Subversion的简称&#xff0c;是一个集中式版本控制系统。与Git不同&#xff0c;SVN没有分布式的特性。在SVN中&#xff0c;项目的代码仓库位于服务器上&#xff0c;团队成员通过向服务器提交和获取代码来实现版本控制。SVN记录了每个…

Leetcode刷题详解—— 目标和

1. 题目链接&#xff1a;494. 目标和 2. 题目描述&#xff1a; 给你一个非负整数数组 nums 和一个整数 target 。 向数组中的每个整数前添加 或 - &#xff0c;然后串联起所有整数&#xff0c;可以构造一个 表达式 &#xff1a; 例如&#xff0c;nums [2, 1] &#xff0c;可…

时间序列预测实战(十二)DLinear模型实现滚动长期预测并可视化预测结果

官方论文地址->官方论文地址 官方代码地址->官方代码地址 个人修改代码->个人修改的代码已经上传CSDN免费下载 一、本文介绍 本文给大家带来是DLinear模型&#xff0c;DLinear是一种用于时间序列预测&#xff08;TSF&#xff09;的简单架构&#xff0c;DLinear的核…

Ansible自动化运维工具及模块

目录 一、Ansible 1.ansible简介 2、ansible的特性 二、ansible的部署 1&#xff09;管理端安装ansible 2&#xff09;配置主机清单 3&#xff09;配置密钥对验证 三、ansible命令块模块 1&#xff09;command模块 2&#xff09;shell模块 3&#xff09;cron模块 4)…

Jdk 1.8 for mac 详细安装教程(含版本切换)

Jdk 1.8 for mac 详细安装教程&#xff08;含版本切换&#xff09; 官网下载链接 https://www.oracle.com/cn/java/technologies/downloads/#java8-mac 一、选择我们需要安装的jdk版本&#xff0c;这里以jdk8为例&#xff0c;下载 macOS 版本&#xff0c;M芯片下载ARM64版本…

数据结构之双向链表

目录 引言 链表的分类 双向链表的结构 双向链表的实现 定义 创建新节点 初始化 打印 尾插 头插 判断链表是否为空 尾删 头删 查找与修改 指定插入 指定删除 销毁 顺序表和双向链表的优缺点分析 源代码 dlist.h dlist.c test.c 引言 数据结构…