lua表,最简单的方法来重载__tostring

Tim*_*ies 2 lua tostring metatable

感谢所有Lua stackoverflow人员,他们讨论了如何自定义打印表.经过多次阅读,我发布以下内容并询问Lua大师....

  • 这是最简单的方法吗?
  • 它太简单了(即以某种方式破坏我不明白)?

请注意以下内容:

  • 允许私有字段(只是用"_"开头)不要打印
  • 不会metatable为每个对象添加大小.

我的方法重写了默认tostring方法.

_tostring =  _tostring or tostring
function tostring(t)
  if type(t) == "table" then 
    status, stuff = pcall(function() return t:s() end) 
    if status then 
      return stuff 
  end end 
  return _tostring(t) 
end
Run Code Online (Sandbox Code Playgroud)

以上是一个有点邪恶(调用pcall ...不是我最自豪的代码,但是,嘿,它的工作原理).

无论如何,现在tostring让方法调用t:s()一个我们可以使用以下自制对象系统定义的对象:

Object={}

function Object:new(o)
   o = o or {} 
   setmetatable(o,self)  
   self.__index = self
   return o
end
Run Code Online (Sandbox Code Playgroud)

这是默认定义:s()- 可以在子类中自定义.

function Object:s()
  -- can be customized in subclasses
  local out,sep="{",":"
  for x,y in pairs(self) do 
    if string.sub(x,1,1) ~= "_" then
      out = out..sep..x.." "..y 
      sep = " :"
  end end  
  return out .. '}'
end
Run Code Online (Sandbox Code Playgroud)

例如

x=Object:new{a=1, _b=2};print(x)
{:a 1}
Run Code Online (Sandbox Code Playgroud)

Col*_*Two 6

这是最简单的方法吗?

到目前为止,没有.最简单的方法是向__tostringmetatable 添加一个函数.

function MyClass:__tostring()
    return "<MyClass: "..self.foo..">"
end
Run Code Online (Sandbox Code Playgroud)

不会为每个对象的元表添加大小.

这不是一个问题.每个类只应存在一个元表.表中一个条目的内存使用量可以忽略不计.

覆盖tostring既丑陋又有潜在危险:如果您(或其他人)正在使用其对象具有s副作用的方法的库,该怎么办?

  • 请注意,Lua 5.1中的`__tostring`不会影响对`lua_tolstring`的调用.这在5.2及更高版本中得到修复 (3认同)