参数函数与LuaJIT FFI的可变参数

有没有比这更有效率的处理 argv 函数的方法?

ffi.cdef [[
  void fooArgv(int argc, const char ** argv, const size_t * argvlen);
]]

local foo = function(...)
  local nargs = select("#", ...)
  local argv = { }
  local argvlen = { }

  for i = 1, nargs do
    local v = tostring( (select(i, ...)) )
    argv[i] = v
    argvlen[i] = #v
  end

  return ffi.C.fooArgv(
      nargs,
      ffi.new("const char * [" .. nargs .. "]", argv),
      ffi.new("const size_t [" .. nargs .. "]", argvlen)
    )
  end
end
点赞
用户1008957
用户1008957

如果foo被多次调用,可以轻松优化函数的底部。每次使用字符串参数调用ffi.new强制LuaJIT运行其C解析器,这不是最优的。函数ffi.typeof可以为给定类型创建一个_构造函数_,然后使用它来替代ffi.new

另外,我认为在循环中使用select函数比创建数组并进行索引要慢。但我并不确定。

所以,这里是我的建议版本:

ffi.cdef [[
  void fooArgv(int argc, const char ** argv, const size_t * argvlen);
]]

local argv_type = ffi.typeof("const char* [?]")
local argvlen_type = ffi.typeof("const size_t [?]")

local foo = function(...)
  local nargs = select("#", ...)
  local argv = { ... }
  local argvlen = { }

  for i = 1, nargs do
    local v = tostring( argv[i] )
    argv[i] = v
    argvlen[i] = #v
  end

  return ffi.C.fooArgv(
      nargs,
      argv_type(nargs, argv),
      argvlen_type(nargs, argvlen)
    )
end
2012-12-23 18:50:12