list – Lua函数在源代码中使用“self”,但没有metamethod允许使用它们

我一直在挖掘Lua的源代码,包括他们网站的C源代码和 Windows上的Lua的lua文件.我找到了一些奇怪的东西,我找不到任何关于他们为什么选择这样做的信息.

字符串库中有一些允许OOP调用的方法,方法是将方法附加到字符串,如下所示:

string.format(s,e1,e2,...)
s:format(e1,...)

所以我挖掘了模块表的源代码,发现像table.remove()这样的函数也允许同样的事情.

这是来自UnorderedArray.lua的源代码:

function add(self,value)
    self[#self + 1] = value
end

function remove(self,index)
    local size = #self
    if index == size then
        self[size] = nil
    elseif (index > 0) and (index < size) then
        self[index],self[size] = self[size],nil
    end
end

这表明函数应该支持冒号方法.当我把表复制到我的新列表中时,看到了这些方法.这是一个使用table.insert作为方法的示例:

function copy(obj,seen) -- Recursive function to copy a table with tables
  if type(obj) ~= 'table' then return obj end
  if seen and seen[obj] then return seen[obj] end

  local s = seen or {}
  local res = setmetatable({},getmetatable(obj))
  s[obj] = res
  for k,v in pairs(obj) do res[copy(k,s)] = copy(v,s) end
  return res
end

function count(list) -- Count a list because #table doesn't work on keyindexed tables
  local sum = 0; for i,v in pairs(list) do sum = sum + 1 end; print("Length: " .. sum)
end

function pts(s) print(tostring(s)) end -- Macro function

local list = {1,2,3}
pts(list.insert) --> nil

pts(table["insert"]) --> function: 0xA682A8
pts(list["insert"]) --> nil

list = copy(_G.table)
pts(table["insert"]) --> function: 0xA682A8
pts(list["insert"]) --> function: 0xA682A8

count(list) --> Length: 9

list:insert(-1,"test")

count(list) --> Length: 10

是Lua 5.1和更新的应该支持像字符串库这样的表方法,但他们决定不实现元方法?

编辑:

我会进一步解释,以便人们理解.
字符串附加了metamethods,您可以在字符串OOP样式上使用.

s = "test"
s:sub(1,1)

但表没有.即使表的源代码中的方法允许使用“自我”功能.所以下面的代码不起作用:

t = {1,3}
t:remove(#t)

该函数在参数中定义了一个self成员(UnorderedArray.lua:25:function remove(self,index)).

您可以使用以下方法找到字符串的元方法:

for i,v in pairs(getmetatable('').__index) do
  print(i,tostring(v))
end

它打印了可用于字符串的所有方法的列表:

sub     function: 0xB4ABC8
upper   function: 0xB4AB08
len     function: 0xB4A110
gfind   function: 0xB4A410
rep     function: 0xB4AD88
find    function: 0xB4A370
match   function: 0xB4AE08
char    function: 0xB4A430
dump    function: 0xB4A310
gmatch  function: 0xB4A410
reverse function: 0xB4AE48
byte    function: 0xB4A170
format  function: 0xB4A0F0
gsub    function: 0xB4A130
lower   function: 0xB4AC28

如果将模块/库表附加到示例中显示的Oka表中,则可以使用表与字符串元方法工作方式相同的方法.

问题是:为什么Lua开发人员默认允许字符串的元方法,但表格甚至不是表格的库,它的方法允许在源代码中使用它?

回答的问题是:它允许模块或程序的开发人员改变程序中所有表的元表,从而导致在程序中使用表时行为与vanilla Lua不同的结果.如果您实现一个数据类型(例如:vectors)并更改该特定类和表的元方法,而不是更改所有Lua的标准表元方法,则会有所不同.这也与运算符重载略有重叠.

解决方法

如果我正确理解你的问题,你会问为什么不可能做到以下几点:

local tab = {}
tab:insert('value')

使用默认的metatable和__index生成表会打破一些关于表的假设.

主要是,空表应该是空的.如果要使用__index元方法查找表来插入,排序等方法,那么它将打破空表不应响应任何成员的假设.

如果您使用表作为缓存或备忘录,这将成为一个问题,您需要检查是否存在“插入”或“排序”字符串(想想任意用户输入).您需要使用rawget来解决首先不需要的问题.

空表也应该是孤儿.这意味着如果没有程序员明确地给予他们关系,他们应该没有任何关系.表是Lua中唯一可用的复杂数据结构,是许多程序的基础.他们需要自由灵活.将它们与表表配对作为默认元表会产生一些不一致.例如,并非所有表都可以使用泛型排序函数 – 类似字典表的奇怪之处.

另外,考虑到你正在使用一个库,并且该库的作者告诉你某个函数返回一个密集的表(即数组),所以你想你可以调用:sort(…)on the退回的表.如果库作者更改了该返回表的元表怎么办?现在你的代码不再有效,并且构建在_:sort(…)范例之上的任何泛型函数都不能接受这些表.

基本上,字符串和表格是两种非常不同的野兽.字符串是不可变的,静态的,并且它们的内容是可预测的.表是可变的,瞬态的,非常难以预测.

在需要时添加它会更容易,而不是将其烘焙到语言中.一个非常简单的功能:

local meta = { __index = table }

_G.T = function (tab)
    if tab ~= nil then
        local tab_t = type(tab)

        if tab_t ~= 'table' then
            error(("`table' expected,got: `%s'"):format(tab_t),0)
        end
    end

    return setmetatable(tab or {},meta)
end

现在,只要你想要一个响应表表中的函数的表,只需在它前面添加一个T.

local foo = T {}

foo:insert('bar')

print(#foo) --> 1

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


1.github代码实践源代码是lua脚本语言,下载th之后运行thmain.lua-netTypevgg-savevgg_cifar10/-S0.0001,报错: 试看看安装lua:报错了,参考这篇文章:ubuntu18.04安装lua的步骤以及出现的问题_weixin_41355132的博客-CSDN博客问题解决,安装成功:情况并没有好转,出现相
此文为搬运帖,原帖地址https://www.cnblogs.com/zwywilliam/p/5999924.html前言在看了uwa之前发布的《Unity项目常见Lua解决方案性能比较》,决定动手写一篇关于lua+unity方案的性能优化文。整合lua是目前最强大的unity热更新方案,毕竟这是唯一可以支持ios热更新的办法。然而作
Rime输入法通过定义lua文件,可以实现获取当前时间日期的功能。1.TIMERime是一款可以高度自定义的输入法,相关教程可以查看往期文章,关于时间获取是指输入一个指定关键字,输出当前时间,效果如下(我定义了time关键字):实现如下:①在用户文件夹中新建一个rime.lua文件加入如下代码 ti
localfunctiongenerate_action(params)localscale_action=cc.ScaleTo:create(params.time,params.scale_x,params.scale_y)localfade_action=cc.FadeIn:create(params.time)returncc.Spawn:create(scale_action,fade_action)end
2022年1月11日13:57:45 官方:https://opm.openresty.org/官方文档:https://opm.openresty.org/docs#table-of-contents为什么建议使用opm不建议使用luarocks?http://openresty.org/cn/using-luarocks.html官方解释:请注意!LuaRocks并不是OpenResty官方推荐的装包方式。LuaRoc
在Lua中的table(表),就像c#中的HashMap(哈希表),key和value一一对应。元表:table的一个操作的拓展,里面包含关联了对应的方法,元方法就是其中一个。元方法:当你通过键来访问table的时候,如果这个键没有值,那么Lua就会寻找该table的metatable(假定有metatable)中的__index键。如果__inde
表排序:table.sort(list[,comp])参数list:指定表,可选参数comp:排序函数,无参数时通常按升序排序。排序函数针对表中连续的序列,其间不可以存在空洞或nil,排序函数需要两个形参(对应表中每次参加比较的两个数据),需要一个比较两个形参表达式的返回值,不能含有等于关系,例如>=,<=,==。do
一、安装lua环境1.1安装依赖包[root@centos7~]#yuminstallgccreadline-devel1.2下线lua源码包并解压[root@centos7~]#wgethttp://www.lua.org/ftp/lua-5.3.5.tar.gz[root@centos7~]#tarxvflua-5.3.5.tar.gz-C/usr/local/src1.3进行编译[root@centos7~]
官网OpenResty® 是一个基于 Nginx 与Lua的高性能Web平台,其内部集成了大量精良的Lua库、第三方模块以及大多数的依赖项。用于方便地搭建能够处理超高并发、扩展性极高的动态Web应用、Web服务和动态网关。OpenResty® 通过汇聚各种设计精良的 Nginx 模块(主要由
表参考《lua程序设计》可以认为,表是一种动态分配的对象,程序只能操作指向表的引用(或指针)。除此以外,Lua语言不会进行隐藏的拷贝(hiddencopies)或创建新的表--创建表a={}--创建空表k="x"a[k]=10--键“x”值10a[20]="great"--键20值“great”print(a["x"])-->10
https://github.com/galenho/crossover.git一个跨平台的lua游戏服务器开发框架,该框架采用多线程并发来处理消息,开发者只需要调用相应的接口函数并绑定相应的回调函数即可,在逻辑层表现为单线程的开发模式,使开发者易用,易调试,易维护,易扩展,同时拥有快速的响应能力。   框架使用面
参考链接:https://www.runoob.com/lua/lua-metatables.htmlhttps://www.jianshu.com/p/cb945e7073a3 元表是一个table,可以让我们改变table的行为,每个行为有对应的元方法例如,对table进行设置键值,查找键值,运算等,就会触发对应的元方法1--__index:table被访问时,如果找不到这
https://github.com/yuin/gopher-luahttps://github.com/yuin/gopher-lua Lua5.1ReferenceManual-contentshttp://www.lua.org/manual/5.1/ go中使用luapackagemainimport( lua"github.com/yuin/gopher-lua")funcmain(){ l:=lua.NewState() d
编译问题不要留到运行时才跑出来啊。早上9:00-中午3:00,6个小时,服了自己了。 写了一个测试,springboot+redis+lua执行到redisTemplate.execute(redisScript,idList)的时候一直报错,integer无法转换为string。我一直以为是lua脚本写错了,翻文档翻过来又翻过去,写法变了又变,还是解
        。。是字符串连接符,字典用=号连接,  注意fordoend都是连一起,  注意ifthen,  如果local在函数里,是可以访问,非local出了函数一样能用,  doend代码块也是一样,    注意点号表示,只能key是字符串,  注意括号不是必须
C语言与Lua之间的相互调用详解写一个C调用Lua的Demo编译运行C语言调用Lua编译问题总结正确的编译命令问题1:缺少-lm参数问题2:缺少-ldl参数​1、为什么会出现undefinedreferenceto‘xxxxx’错误?​2、-l参数和-L参数写一个C调用Lua的Demo编译运行add.c内容//你需要
1、动态输出打开E:\study\openresty\openresty-1.19.9.1-win64目录下的confginx.conf文件在server中增加一下代码 location/hello{ default_typetext/html; content_by_lua'ngx.say("<p>hello,world</p>")'; }运行后,效果如下图localhost
参见:lipp/lua-websockets:WebsocketsforLua.(github.com)github网址可能需手动转换lipp.github.com/lua-websockets/>github.com/lipp/lua-websocketswebsockets为底层的类似于TCP、UDP的socket(实现上基于更底层的socket),不同于上层的webserver服务端(Service)需并行地支持多
lua发送消息到rabbitmq,我们选择类库lua-resty-rabbitmqstomp 来完成这个任务。类库安装:进入nginx.conf中 lua_package_path 中对应的目录下的resty目录(没有则创建),执行:wget-chttps:/aw.githubusercontent.com/wingify/lua-resty-rabbitmqstomp/master/libes
1Lua介绍Lua是一门以其性能著称的脚本语言,被广泛应用在很多方面。Lua一般用于嵌入式应用,现在越来越多应用于游戏当中,魔兽世界,愤怒的小鸟都有用到。优势Lua极易嵌入到其他程序,可当做一种配置语言。提升应用性能,比如:游戏脚本,nginx,wireshark的脚本兼容性强,可以直接使用C