百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

【LUA】只需花费你半天时间

wptr33 2025-01-21 21:56 27 浏览

前言:有一段时间使用OpenResty写Waf防护模块的时候使用到了Lua。Lua 是一种轻量小巧的脚本语言,用标准C语言编写并以源代码形式开放, 其设计目的是为了嵌入应用程序中,从而为应用程序提供灵活的扩展和定制功能。

学习Lua代码,从变量到跑路

x=1    --全局变量  
local x=1 --局部变量

function a()  
b=2     --全局变量  
local c=2 --局部变量  
end

print(b,c)    --  2,nil

local _M = {}    --空tabel 也叫空数组  
_M["key"] = "value" --填充值  
--给tabel增加方法
_M.Find = function()  
print("local")  
end

d,d2 = 2,3 -- 定义值,多个  
d,d2 = d2,d   --swap交换值  
print(d,d2)  --3,2

--条件语句
if true then  
print(xxx)  
end

if true then  
print(xx)  
else  
if false then  
print(x)  
end  
end

--遍历tabel
for k,v in ipairs(_M) do  
print(k,v)  
end  
--ipairs和pairs都是的迭代器,区别,
--ipairs遇到tabel内容为nil的时候,终止循环
--注意:lua迭代器下标是从1开始
for k,v in pairs(_M) do  
print(k,v)  
end

--循环
--x=初始值,最大值,步长值     步长值代表每次递增多少数
for x=1,10,3 do  
print(x)  
end  
--while循环
local a = 10  
while(a<100)  
do  
a = a+10  
print(a)  
end

--repeat-until循环,先执行,后判断,类似语言do---while
local b = 10  
repeat  
print(b)  
b = b+1  
until(b>15)    --当b大于15的时候结束循环

--函数定义,系统默认是全局
function a() do  
print("all in")  
end  
--局部函数使用local,也支持向tabel添加方法
local func = function()  
print("local")  
end  
--可变参数,接受未知个参数
funciton args(...)  
local result = 0  
---将参数写入tabel
local arg = {...}  
for k,v = ipairs(arg) do  
print(k,v)  
end  
--#arg代表统计有多少个参数
print("参数总数:",#arg)  
end  
--Demo(当传入为nil参数的时候,是不算个数):
function fun(...)  
    local x={...}
    print(#x)
end  
fun(1,2,3,4,5,nil)   --5  
fun(1,2,3,4,5,0)     --6


---#xx 统计坑,取决于最大的索引值,如果有越标行为,则按越标前一位计算总数
local xx = {}  
xx[1] = 2  
xx[2] = 3  
print(#xx)  
local xx = {}  
xx[1] = 2  
xx[8] = 3  
print(#xx)

--字符串
local x = "aaaaa"  
local x = ’aaaaa‘  
local x = [[  
一组模板数据
]]
--字符串连接 ..
local c = x..b


--模块与包
--定义a.lua文件
a = {}  
a.constant = "常量"  
a.func1 = function()  
print("a模块 1方法")  
end  
a.func2 = function()  
print("a模块 2方法")  
end  
return a  
--定义b.lua文件
--require("a")
require("a")  
a.func1()  
--local m = require("a")
m.func1()

--lua加载c库
local path = "/usr/local/lua/lib/libluasocket.so"  
-- 或者 path = "C:\\windows\\luasocket.dll",这是 Window 平台下
local f = assert(loadlib(path, "luaopen_socket"))  
f()  -- 真正打开库


--协同程序
function foo (a)  
    print("foo 函数输出", a)
    return coroutine.yield(2 * a) -- 返回  2*a 的值
end  
co = coroutine.create(function (a , b)  
    print("第一次协同程序执行输出", a, b) -- co-body 1 10
    local r = foo(a + 1)     
    print("第二次协同程序执行输出", r)
    local r, s = coroutine.yield(a + b, a - b)  -- a,b的值为第一次调用协同程序时传入     
    print("第三次协同程序执行输出", r, s)
    return b, "结束协同程序"                   -- b的值为第二次调用协同程序时传入
end)  
print("main", coroutine.resume(co, 1, 10)) -- true, 4  
print("--分割线----")  
print("main", coroutine.resume(co, "r")) -- true 11 -9  
print("---分割线---")  
print("main", coroutine.resume(co, "x", "y")) -- true 10 end  
print("---分割线---")  
print("main", coroutine.resume(co, "x", "y")) -- cannot resume dead coroutine  
print("---分割线---")  
--[[
第一次运行之后,挂起yield
第二运行,先执行上次的yield输出,再执行本次调用
第三次运行,执行上一次yield输出,再执行本次
结束协同程序
第四次就提示dead了
]]


--文件操作,基于io类
--打开
file = io.open("文件名","打开方式") --r w a r+ w+ a+ b  
--读取
io.input(file)  
io.read()  
--写入
io.output(file)  
io.write("hhhhh")  
--关闭
io.close(file)


--面向对象
A = {t=0}  
A.func1 = function()  
print(A.t)  
end

--继承
B = {area=0}  
--基础类
B:new =function(o,p2)  
o = o or {}  
setmetatable(o, self)  
self.__index = self  
side = side or 0  
self.area = side*side;  
return o  
end  
-- 基础类方法 printArea
function B:printArea()  
  print("面积为 ",self.area)
end

-- 创建对象
myshape = Shape:new(nil,10)  
myshape:printArea()

Square = Shape:new()  
-- 派生类方法 new
function Square:new (o,side)  
  o = o or Shape:new(o,side)
  setmetatable(o, self)
  self.__index = self
  return o
end

-- 派生类方法 printArea
function Square:printArea ()  
  print("正方形面积为 ",self.area)
end

-- 创建对象
mysquare = Square:new(nil,10)  
mysquare:printArea()

Rectangle = Shape:new()  
-- 派生类方法 new
function Rectangle:new (o,length,breadth)  
  o = o or Shape:new(o)
  setmetatable(o, self)
  self.__index = self
  self.area = length * breadth
  return o
end

-- 派生类方法 printArea
function Rectangle:printArea ()  
  print("矩形面积为 ",self.area)
end

-- 创建对象
myrectangle = Rectangle:new(nil,10,20)  
myrectangle:printArea()

总结

如果你有用到openresty作一个补丁包开发,建议使用lua脚本方式,随然c也可以实现...


前言:有一段时间使用OpenResty写Waf防护模块的时候使用到了Lua。Lua 是一种轻量小巧的脚本语言,用标准C语言编写并以源代码形式开放, 其设计目的是为了嵌入应用程序中,从而为应用程序提供灵活的扩展和定制功能。

学习Lua代码,从变量到跑路

x=1    --全局变量  
local x=1 --局部变量

function a()  
b=2     --全局变量  
local c=2 --局部变量  
end

print(b,c)    --  2,nil

local _M = {}    --空tabel 也叫空数组  
_M["key"] = "value" --填充值  
--给tabel增加方法
_M.Find = function()  
print("local")  
end

d,d2 = 2,3 -- 定义值,多个  
d,d2 = d2,d   --swap交换值  
print(d,d2)  --3,2

--条件语句
if true then  
print(xxx)  
end

if true then  
print(xx)  
else  
if false then  
print(x)  
end  
end

--遍历tabel
for k,v in ipairs(_M) do  
print(k,v)  
end  
--ipairs和pairs都是的迭代器,区别,
--ipairs遇到tabel内容为nil的时候,终止循环
--注意:lua迭代器下标是从1开始
for k,v in pairs(_M) do  
print(k,v)  
end

--循环
--x=初始值,最大值,步长值     步长值代表每次递增多少数
for x=1,10,3 do  
print(x)  
end  
--while循环
local a = 10  
while(a<100)  
do  
a = a+10  
print(a)  
end

--repeat-until循环,先执行,后判断,类似语言do---while
local b = 10  
repeat  
print(b)  
b = b+1  
until(b>15)    --当b大于15的时候结束循环

--函数定义,系统默认是全局
function a() do  
print("all in")  
end  
--局部函数使用local,也支持向tabel添加方法
local func = function()  
print("local")  
end  
--可变参数,接受未知个参数
funciton args(...)  
local result = 0  
---将参数写入tabel
local arg = {...}  
for k,v = ipairs(arg) do  
print(k,v)  
end  
--#arg代表统计有多少个参数
print("参数总数:",#arg)  
end  
--Demo(当传入为nil参数的时候,是不算个数):
function fun(...)  
    local x={...}
    print(#x)
end  
fun(1,2,3,4,5,nil)   --5  
fun(1,2,3,4,5,0)     --6


---#xx 统计坑,取决于最大的索引值,如果有越标行为,则按越标前一位计算总数
local xx = {}  
xx[1] = 2  
xx[2] = 3  
print(#xx)  
local xx = {}  
xx[1] = 2  
xx[8] = 3  
print(#xx)

--字符串
local x = "aaaaa"  
local x = ’aaaaa‘  
local x = [[  
一组模板数据
]]
--字符串连接 ..
local c = x..b


--模块与包
--定义a.lua文件
a = {}  
a.constant = "常量"  
a.func1 = function()  
print("a模块 1方法")  
end  
a.func2 = function()  
print("a模块 2方法")  
end  
return a  
--定义b.lua文件
--require("a")
require("a")  
a.func1()  
--local m = require("a")
m.func1()

--lua加载c库
local path = "/usr/local/lua/lib/libluasocket.so"  
-- 或者 path = "C:\\windows\\luasocket.dll",这是 Window 平台下
local f = assert(loadlib(path, "luaopen_socket"))  
f()  -- 真正打开库


--协同程序
function foo (a)  
    print("foo 函数输出", a)
    return coroutine.yield(2 * a) -- 返回  2*a 的值
end  
co = coroutine.create(function (a , b)  
    print("第一次协同程序执行输出", a, b) -- co-body 1 10
    local r = foo(a + 1)     
    print("第二次协同程序执行输出", r)
    local r, s = coroutine.yield(a + b, a - b)  -- a,b的值为第一次调用协同程序时传入     
    print("第三次协同程序执行输出", r, s)
    return b, "结束协同程序"                   -- b的值为第二次调用协同程序时传入
end)  
print("main", coroutine.resume(co, 1, 10)) -- true, 4  
print("--分割线----")  
print("main", coroutine.resume(co, "r")) -- true 11 -9  
print("---分割线---")  
print("main", coroutine.resume(co, "x", "y")) -- true 10 end  
print("---分割线---")  
print("main", coroutine.resume(co, "x", "y")) -- cannot resume dead coroutine  
print("---分割线---")  
--[[
第一次运行之后,挂起yield
第二运行,先执行上次的yield输出,再执行本次调用
第三次运行,执行上一次yield输出,再执行本次
结束协同程序
第四次就提示dead了
]]


--文件操作,基于io类
--打开
file = io.open("文件名","打开方式") --r w a r+ w+ a+ b  
--读取
io.input(file)  
io.read()  
--写入
io.output(file)  
io.write("hhhhh")  
--关闭
io.close(file)


--面向对象
A = {t=0}  
A.func1 = function()  
print(A.t)  
end

--继承
B = {area=0}  
--基础类
B:new =function(o,p2)  
o = o or {}  
setmetatable(o, self)  
self.__index = self  
side = side or 0  
self.area = side*side;  
return o  
end  
-- 基础类方法 printArea
function B:printArea()  
  print("面积为 ",self.area)
end

-- 创建对象
myshape = Shape:new(nil,10)  
myshape:printArea()

Square = Shape:new()  
-- 派生类方法 new
function Square:new (o,side)  
  o = o or Shape:new(o,side)
  setmetatable(o, self)
  self.__index = self
  return o
end

-- 派生类方法 printArea
function Square:printArea ()  
  print("正方形面积为 ",self.area)
end

-- 创建对象
mysquare = Square:new(nil,10)  
mysquare:printArea()

Rectangle = Shape:new()  
-- 派生类方法 new
function Rectangle:new (o,length,breadth)  
  o = o or Shape:new(o)
  setmetatable(o, self)
  self.__index = self
  self.area = length * breadth
  return o
end

-- 派生类方法 printArea
function Rectangle:printArea ()  
  print("矩形面积为 ",self.area)
end

-- 创建对象
myrectangle = Rectangle:new(nil,10,20)  
myrectangle:printArea()

总结

如果你有用到openresty作一个补丁包开发,建议使用lua脚本方式,随然c也可以实现...


相关推荐

MySQL进阶五之自动读写分离mysql-proxy

自动读写分离目前,大量现网用户的业务场景中存在读多写少、业务负载无法预测等情况,在有大量读请求的应用场景下,单个实例可能无法承受读取压力,甚至会对业务产生影响。为了实现读取能力的弹性扩展,分担数据库压...

Postgres vs MySQL_vs2022连接mysql数据库

...

3分钟短文 | Laravel SQL筛选两个日期之间的记录,怎么写?

引言今天说一个细分的需求,在模型中,或者使用laravel提供的EloquentORM功能,构造查询语句时,返回位于两个指定的日期之间的条目。应该怎么写?本文通过几个例子,为大家梳理一下。学习时...

一文由浅入深带你完全掌握MySQL的锁机制原理与应用

本文将跟大家聊聊InnoDB的锁。本文比较长,包括一条SQL是如何加锁的,一些加锁规则、如何分析和解决死锁问题等内容,建议耐心读完,肯定对大家有帮助的。为什么需要加锁呢?...

验证Mysql中联合索引的最左匹配原则

后端面试中一定是必问mysql的,在以往的面试中好几个面试官都反馈我Mysql基础不行,今天来着重复习一下自己的弱点知识。在Mysql调优中索引优化又是非常重要的方法,不管公司的大小只要后端项目中用到...

MySQL索引解析(联合索引/最左前缀/覆盖索引/索引下推)

目录1.索引基础...

你会看 MySQL 的执行计划(EXPLAIN)吗?

SQL执行太慢怎么办?我们通常会使用EXPLAIN命令来查看SQL的执行计划,然后根据执行计划找出问题所在并进行优化。用法简介...

MySQL 从入门到精通(四)之索引结构

索引概述索引(index),是帮助MySQL高效获取数据的数据结构(有序),在数据之外,数据库系统还维护者满足特定查询算法的数据结构,这些数据结构以某种方式引用(指向)数据,这样就可以在这些数据结构...

mysql总结——面试中最常问到的知识点

mysql作为开源数据库中的榜一大哥,一直是面试官们考察的重中之重。今天,我们来总结一下mysql的知识点,供大家复习参照,看完这些知识点,再加上一些边角细节,基本上能够应付大多mysql相关面试了(...

mysql总结——面试中最常问到的知识点(2)

首先我们回顾一下上篇内容,主要复习了索引,事务,锁,以及SQL优化的工具。本篇文章接着写后面的内容。性能优化索引优化,SQL中索引的相关优化主要有以下几个方面:最好是全匹配。如果是联合索引的话,遵循最...

MySQL基础全知全解!超详细无废话!轻松上手~

本期内容提醒:全篇2300+字,篇幅较长,可搭配饭菜一同“食”用,全篇无废话(除了这句),干货满满,可收藏供后期反复观看。注:MySQL中语法不区分大小写,本篇中...

深入剖析 MySQL 中的锁机制原理_mysql 锁详解

在互联网软件开发领域,MySQL作为一款广泛应用的关系型数据库管理系统,其锁机制在保障数据一致性和实现并发控制方面扮演着举足轻重的角色。对于互联网软件开发人员而言,深入理解MySQL的锁机制原理...

Java 与 MySQL 性能优化:MySQL分区表设计与性能优化全解析

引言在数据库管理领域,随着数据量的不断增长,如何高效地管理和操作数据成为了一个关键问题。MySQL分区表作为一种有效的数据管理技术,能够将大型表划分为多个更小、更易管理的分区,从而提升数据库的性能和可...

MySQL基础篇:DQL数据查询操作_mysql 查

一、基础查询DQL基础查询语法SELECT字段列表FROM表名列表WHERE条件列表GROUPBY分组字段列表HAVING分组后条件列表ORDERBY排序字段列表LIMIT...

MySql:索引的基本使用_mysql索引的使用和原理

一、索引基础概念1.什么是索引?索引是数据库表的特殊数据结构(通常是B+树),用于...