0

有没有比这更有效的方法来处理 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
4

1 回答 1

2

如果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-23T18:50:12.730 回答