接受可变参数并返回可变参数的lambda函数Lua
2017-8-18 0:15:46
收藏:0
阅读:175
评论:3
如果我想写一个接受可变数目参数并返回一个接受可变数目参数的函数,我将遇到模糊的...,例如
function bind(func, ...) return function(...) func(..., ...) end end
点赞
用户2328287
你可以尝试使用vararg库。该库还可以处理参数中的nil。
va = require "vararg"
function bind(func, ...)
local args = va(...)
return function(...)
func(va.concat(args, va(...)))
end
end
bind(print, 1, nil, 2)(3, nil, 5)
2017-04-20 09:42:57
用户8261525
这种方法类似于 Rici Lake 的 Partial,使用了记忆化的辅助函数而不是表格打包/解包。它有大约 2 倍的性能优势,并且具有较低的内存使用率。
local fmt,cat,pack = string.format,table.concat,table.pack
local function args(n, Prefix)
local as, prefix = {}, Prefix or '_'
local step, from, to = n < 0 and -1 or 1, n < 0 and -n or 1, n < 0 and 1 or n
for i = from, to, step do as[1+#as] = prefix .. i end
return function(sep) return cat(as, sep or ',') end
end
local function paramsCat(...)
local r, p = {}, pack(...)
for i = 1, p.n do if p[i]:len() > 0 then r[1+#r] = p[i] end end
return cat(r, ',')
end
local bind = setmetatable({}, {
__call = function(self, f, ...)
local narg = select("#",...)
if not self[narg] then
local a = args(narg)()
local b = '...'
local src = fmt([[
return function(%s) -- _1,_2
return function(fn)
return function(...)
return fn(%s) -- _1,_2,...
end
end
end]],a, paramsCat(a,b))
local fn = load(src,'_')()
self[narg] = fn
end
return self[narg](...)(f)
end
})
通过轻微修改,我们可以扩展绑定到第 n 个参数开始,
local bindn = setmetatable({}, {
__call = function(self, n, f, ...)
local narg = select("#",...)
if type(n) ~= 'number' then -- shifted, n is the function now
if f ~= nil or narg > 0 then
return self(0, n, f, ...)
else
return self(0, n)
end
end
self[n] = self[n] or {}
if not self[n][narg] then
local a = args(n)()
local b = args(narg,'b')()
local c = '...'
local src = fmt([[
return function(%s) -- b1,b2
return function(fn)
return function(%s) -- _1,_2,_3,...
return fn(%s) -- _1,_2,_3,b1,b2,...
end
end
end]],b, paramsCat(a,c), paramsCat(a,b,c))
local fn = load(src,'_')()
self[n][narg] = fn
end
return self[n][narg](...)(f)
end
})
local dp = bindn(2, print, 'debug-', 'print:')
dp(1, 2, 3, 4, 5) --> 1 2 debug- print: 3 4 5
dp = bindn(print, 'debug-', 'print:')
dp(1, 2, 3, 4, 5) --> debug- print: 1 2 3 4 5
string.Bytes = bindn(1, string.byte, 1, -1)
print(("Test"):Bytes()) --> 84 101 115 116
2017-08-03 16:08:14
评论区的留言会收到邮件通知哦~
推荐文章
- Lua 虚拟机加密load(string.dump(function)) 后执行失败问题如何解决
- 我想创建一个 Nginx 规则,禁止访问
- 如何将两个不同的lua文件合成一个 东西有点长 大佬请耐心看完 我是小白研究几天了都没搞定
- 如何在roblox studio中1:1导入真实世界的地形?
- 求解,lua_resume的第二次调用继续执行协程问题。
- 【上海普陀区】内向猫网络招募【Skynet游戏框架Lua后端程序员】
- SF爱好求教:如何用lua实现游戏内调用数据库函数实现账号密码注册?
- Lua实现网站后台开发
- LUA错误显式返回,社区常见的规约是怎么样的
- lua5.3下载库失败
- 请问如何实现文本框内容和某个网页搜索框内容连接,并把网页输出来的结果反馈到另外一个文本框上
- lua lanes多线程使用
- 一个kv数据库
- openresty 有没有比较轻量的 docker 镜像
- 想问一下,有大佬用过luacurl吗
- 在Lua执行过程中使用Load函数出现问题
- 为什么 neovim 里没有显示一些特殊字符?
- Lua比较两个表的值(不考虑键的顺序)
- 有个lua简单的项目,外包,有意者加微信 liuheng600456详谈,最好在成都
- 如何在 Visual Studio 2022 中运行 Lua 代码?

首先,你漏掉了一个可绑定函数关闭的结束符。
如果有歧义,只需通过使用不同的名称解决它们。
function bind(func, ...) return function(...) func(..., ...) end end如果我们像这样测试你的代码:
bind(print,“a”,“b”,“c”)(1,2,3)你会得到输出:
如果在函数参数列表中有...或任何其他名称,则该变量将是该函数范围内的本地变量。它将优先于任何其他具有相同名称的变量在更高的范围。因此,匿名函数中的...与函数绑定中的...无关。
要解决此问题,您可以简单地执行以下操作
function bind(func, ...) local a = table.pack(...) return function(...) func(table.unpack(a, 1, a.n), ...) end end现在调用bind(print,“a”,“b”,“c”)(1,2,3)将输出:
要了解发生了什么,请阅读此部分:[https://www.lua.org/manual/5.3/manual.html#3.4.11](https://www.lua.org/manual/5.3/manual.html#3.4.11)
(当然还有Lua的其他手册)
因此,即使...是两个不同的列表,像func(...,...)这样的东西也永远不会起作用。
为避免这种情况,您必须连接两个参数列表。
function bind(func, ...) local args1 = table.pack(...) return function(...) local args2 = table.pack(...) for i = 1,args 2.n do args1[args1.n+i] = args2[i] end args1.n = args1.n + args2.n func(table.unpack(args1, 1, args1.n)) end endbind(print,“a”,nil,“c”)(1,nil,3)
最后,我们得到了所需的输出:
但是我相信您可以想到更好的方法来实现您的目标,而无需连接各种varargs。