首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >用于在表中创建变量的Lua循环

用于在表中创建变量的Lua循环
EN

Stack Overflow用户
提问于 2014-01-04 00:22:50
回答 2查看 1.1K关注 0票数 4

循环有问题,Lua中的表是带有变量knx的表(现在它是静态的)

代码语言:javascript
复制
regTable = {
      { RegEddr=3027, count=2, regType="float", knx="1/1/1"},
      { RegEddr=3029, count=2, regType="float", knx="1/1/2"},
      { RegEddr=3031, count=2, regType="float", knx="1/1/3"},
      { RegEddr=2999, count=2, regType="float", knx="1/1/4"},
      { RegEddr=3001, count=2, regType="float", knx="1/1/5"},
      { RegEddr=3003, count=2, regType="float", knx="1/1/6"},
      { RegEddr=3109, count=2, regType="float", knx="1/1/7"},
      { RegEddr=3083, count=2, regType="float", knx="1/1/8"},
      { RegEddr=3059, count=2, regType="float", knx="1/1/9"},
      { RegEddr=3203, count=4, regType="int64", knx="1/1/10"},
    }

    function readRegisters()

    for idx, register in pairs(regTable) do
      if register.regType=="int" then
        valueInt = mb:readregisters(register.RegEddr)
        grp.write(register.knx, valueInt)




            elseif register.regType=="float" then
                     value1, value2 = mb:readregisters(register.RegEddr,register.count)

            if value1 then
                     valueFloat = bit.lshift(value1, 16) + value2
                     valueFloat = lmcore.inttohex(valueFloat, 4)
                     valueFloat = knxdatatype.decode(valueFloat, dt.float32)
            grp.write(register.knx, valueFloat)
               end

           elseif register.regType=="int64" then
          valueInt1, valueInt2, valueInt3, valueInt4 = mb:readregisters(register.RegEddr,register.count)
          if valueInt4 then
            valueInt64 = valueInt4
                log(valueInt64)
                grp.write(register.knx, valueInt64)
            end




       end

      end     --end for

    end --end function

从另一个脚本中,我调用函数readRegisters(),所以我有地址列表,但我不知道用户需要多少地址。如果是10还是100。这就是为什么使用+1步骤的动态列表而不是地址列表最优的原因

代码语言:javascript
复制
1/1/1
1/1/2
...
1/1/255

它可以帮助我如何创建动态添加地址变量knx到这个表?

EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2014-01-04 14:02:06

您需要一个寄存器分配器函数,它考虑到最后一个寄存器的地址和大小。当您请求时,此分配程序将动态创建新寄存器。

代码语言:javascript
复制
local startAddr = 3000
local sizes = {float = 2, int = 2, int64 = 4}
local registers = {}

local function allocRegister(type)
  if sizes[type] == nil then
    error'invalid register type'
  end

  local n = #registers
  local addr

  if n == 0 then -- If this is the first register, use the starting address.
    addr = startAddr
  else -- Determine the next starting address based on the last register's address & size.
    addr = registers[n].addr + registers[n].count
  end

  table.insert(registers, { addr = addr, count = sizes[type], type = type, knx = '1/1/' .. n + 1 })
end

-- Example usage:
allocRegister'float'
allocRegister'int64'
allocRegister'int'
代码语言:javascript
复制
-- Resulting table:
{
  { addr = 3000, count = 2, knx = "1/1/1", type = "float" },
  { addr = 3002, count = 4, knx = "1/1/2", type = "int64" },
  { addr = 3006, count = 2, knx = "1/1/3", type = "int" }
}

您也可以在循环中使用此函数。下面的循环将创建一个与问题中的注册表非常相似的注册表。

代码语言:javascript
复制
for i=1, 9 do allocRegister'float' end
allocRegister'int64'

编辑:下面的代码应该具有足够的说明性,以向您展示如何解决问题。

代码语言:javascript
复制
local sizes = {float = 2, int = 2, int64 = 4}
local registers = {}
local usedSpace = {}

local function allocRegisters(t)
  for i=1, #t do
    local addr, size = t[i].addr, sizes[t[i].type]

    if size == nil then
      error('invalid register type: ' .. t[i].type)
    end

    -- Check if there's free space for this register.
    for j=addr, addr+size-1 do
      if usedSpace[j] then
        error('address already in use: ' .. addr)
      end
    end

    -- Mark the space for this register as used.
    for j=addr, addr+size-1 do
      usedSpace[j] = true
    end

    -- Copy the register into the registers table, setting knx by using the length of the table.
    table.insert(registers, { addr = addr, count = size, type = t[i].type, knx = '1/1/' .. #registers + 1})
  end
end

-- Example usage:
allocRegisters {
  { addr = 3000, type = 'float' },
  { addr = 3003, type = 'int' },
  { addr = 3009, type = 'int64' }
}
票数 1
EN

Stack Overflow用户

发布于 2014-01-04 15:06:04

所以您需要动态地创建regTable,knx是以'1/1/n‘的形式动态生成的,其中n是创建顺序。您可能还希望根据寄存器类型自动计算计数。似乎地址可以按任何顺序分配,但它们必须与寄存器类型大小一致。那么这个怎么样:

代码语言:javascript
复制
local counts = {float = 2, int = 2, int64 = 4}
local regTable = {}

local function newRegister(addr, regType)
  local count = counts[regType]
  if count == nil then
    error 'invalid register type'
  end

  checkAddrVsCount(addr, count) -- verify that addr is ok

  local tableSize = #regTable
  knx = '1/1/' .. (tableSize+1)
  regTable.insert {RegEddr=addr, count=count, regType=regType, knx=knx}
end

checkAddrVsCount(addr, count)检查addr是否有效。例如,

  • 如果regTable包含一个具有RegEddr=addr的条目,则出错(addr已经在表中)
  • 否则,查找与addr最接近的较小的RegEddr条目,并检查所发现条目的类型(而不是新条目的类型)的差距是否至少是计数类型。
票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/20915164

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档