如何在表中表示nil

时间:2016-11-05 18:03:23

标签: lua lua-table

假设我想存储一个元素列表。包括一些零值。值的位置很重要,我需要在给定位置表示列表中缺少值。

这是一个问题:

a = {1,2,3,nil,4}

for k,v in ipairs(a) do
  print(k,v)
end

print(a[4]) 
print(a[5])

for循环只会打印元素1,2和3.它会停止为零。第一个print语句打印nil,但我不确定它是否实际存储在表中。 (谁知道?)第二个打印语句打印4 - 正如预期的那样。

所以这就是问题:如何表示表中的元素列表,并有效地迭代它们?鉴于上述条件,例如这个职位很重要,有些职位空缺了#34;换句话说:没有价值,但在那个位置缺少那个价值就有意义。

5 个答案:

答案 0 :(得分:4)

Lua表可用于创建任何抽象数据结构,在您指定您需要“列表”的情况下。 Lua表是一种数据结构,它将基于数字索引的访问与key:value访问相结合。

根据您的示例,您正在使用表的数字索引功能,您可以通过这些值进行迭代(使用ipairs())。由于数字索引在第一个nil条目处停止,因此您将无法将nil放入表中。表中的其余值存储为键:值对。

有几种解决方法,但这取决于你想要列表中的nil的原因。最简单的方法是使用字符串“nil”而不是本机数据类型nil。

a = {1, 2, 3, "nil", 4}

for k,v in ipairs(a) do
  print(k,v)
end

此代码的结果是:

1   1 
2   2
3   3
4   nil
5   4

由于Lua实现字符串的方式,与字符串“nil”相比,与本机类型nil相比,没有性能损失。

在Lua编程中,第5章表中讨论了数组中“漏洞”(由nil引起)的问题。 Roberto Ierusalimschy建议跟踪阵列的大小以避免漏洞问题。

以下代码显示了一种用于跟踪列表大小的面向对象方法。这个主题有很多可能的变化。

function makeList(...)
  local list = table.pack(...)

  list.length = 
    function(self) return self.n 
    end

  list.append = 
    function(self, value)
      self.n = self.n + 1
      self[self.n] = value
    end

  list.print = 
    function(self)
      for i = 1, self.n do print(i, self[i]) end
    end

  return list
end

a = makeList(1, 2, 3, nil, 4)
a:append(5)

a:print()

print(a:length())

结果是:

1   1
2   2
3   3
4   nil
5   4
6   5
6

请注意,函数table.pack会创建一个字段'n',其中包含正确数量的项,即使存在'nil'也是如此。有关完整说明,请参见PIL第6.2章,Variadic函数。

答案 1 :(得分:4)

这是模块“null.lua”

local function null(...)
   local t, n = {...}, select('#', ...)
   for k = 1, n do
      local v = t[k]
      if     v == null then t[k] = nil
      elseif v == nil  then t[k] = null
      end
   end
   return (table.unpack or unpack)(t, 1, n)
end
_G.null = null

使用null()作为编码器和解码器

require("null")

a = {null(1,2,3,nil,4)}
-- the same could be done element-by-element
-- a = {null(1),null(2),null(3),null(nil),null(4)}

for k,v in ipairs(a) do
   v = null(v)
   print(k,v)
end

print(null(a[4]))
print(null(a[5]))

答案 2 :(得分:1)

不要只是一起破解某些东西,为此编写自己的数据结构。如果你过载" ipairs(通过编写适当的迭代器),您可以将其用作表格:

function create(...)
    local t = table.pack(...)
    local self = {
        num = t.n,
        elements = { ... }
    }
    return self
end

function elements(t)
    local f = function(s, i)
        i = i + 1
        if i <= s.num then
            return i, s.elements[i]
        end
    end
    return f, t, 0
end

local seq = create(1, 2, nil, 3)

print(seq.num)
for i, e in elements(seq) do
    print(i, e)
end
-- results:
-- 4
-- 1    1
-- 2    2
-- 3    nil
-- 4    3

你可以知道为这个结构定义一个metatable并让它使用自己的ipairs,所以你甚至不必更改名称。

答案 3 :(得分:1)

答案很简单,这些&#34;解决方法是&#34;建议肯定是矫枉过正。只要记录表中的项目数量,就可以随时跟踪(注意:不要使用#,你也要手动跟踪以处理nil值)并使用数字for循环来迭代它。 / p>

答案 4 :(得分:0)

好吧,你不能在没有问题的情况下将nil存储在表格中。

这里最简单的解决方案是介绍您自己的独特价值。

local mynil = {} -- every new table is unique!

a = {1,2,3,mynil,4}

for k,v in ipairs(a) do
  if (v == mynil) then
    v = nil
  end
  print(k,v)
end

不再存在可能存储在表中的“nil”字符串的问题,小问题是另外一个比较。 ipairs或任何其他迭代器将显示存在mynil值的密钥。这意味着您可以将mynil密钥存在与缺少密钥=nil分开。

P.S。如果您想转移列表,可以考虑table.remove(list, key)功能。