3

想象一下下面的代码:

Mytable={}
print(Mytable)

打印类似Table: 12345. 如何在不弄乱tostring' 的返回值的情况下从 Lua 获取“地址”部分,更重要的是,如何取回表格?

在代码中:

addr=table2address(Mytable)
-- type(addr) is number, addr is 12345
Othertable=address2table(addr)
-- type(Othertable) is table, Othertable==Mytable is true (same reference)

有没有办法在 Lua 中实现这两个功能?如果没有,(如何)我在 C 中做到这一点?

编辑:table2address可以通过切断Table:from来完成tostring(Mytable),但前提__tostring是未定义元方法,所以我想避免这种情况。

4

1 回答 1

4

一个简单的实现满足您的所有标准,但一个:

function table2address(Mytable) return Mytable end
function address2table(addr) return addr end

演示:

> Mytable={}
> print(Mytable)
table: 0x7fe511c0a190
> addr = table2address(Mytable)
> Othertable=address2table(addr)
> =type(Othertable)
table
> print(Othertable==Mytable)
true

稍微复杂一点的实现符合您的所有标准:

t2at = {}

function table2address(Mytable) 
  local addr = t2at[Mytable]
  if addr == nil then
    addr = #t2at + 1
    t2at[Mytable] = addr
    t2at[addr] = Mytable
  end
  return addr
end

function address2table(addr)
  return t2at[addr]
end

演示:

> Mytable={}
> addr = table2address(Mytable)
> Othertable=address2table(addr)
> =type(Othertable)
table
> print(Othertable==Mytable)
true
> =type(addr)
number

那么,为什么地址对您很重要?

在像 Lua 这样的垃圾收集语言中,只能保存对对象的引用,而不是地址。[当前的实现可能会或可能不会在 GC 期间移动对象,但除了userdataLua 声明之外,Lua 有权移动任何东西。]

附录

回复:“地址永远不会随机化(在 2 个新的交互式 lua 实例中尝试 print({}))”

e$ lua
Lua 5.2.2  Copyright (C) 1994-2013 Lua.org, PUC-Rio
> print({})
table: 0x7fdaca4098c0
> ^D
e$ lua
Lua 5.2.2  Copyright (C) 1994-2013 Lua.org, PUC-Rio
> print({})
table: 0x7fb02a4098c0
> ^D
e$ 

Re: 确实需要物理地址

看看luaL_tolstring实现 print 胆量的函数;它有(在 Lua 5.2.2 中):

  default:
    lua_pushfstring(L, "%s: %p", luaL_typename(L, idx),
                                        lua_topointer(L, idx));
    break;

那么,lua_topointer(L, idx)您需要获取表格地址的功能。

于 2013-07-20T14:56:01.797 回答