【发布时间】:2020-09-05 10:57:34
【问题描述】:
假设我有一个cdata 变量(FFI 查询的结果)。然后我将其字段复制到 Lua 表中。哪个更快:访问 Lua 表或原始 cdata 变量的字段?
【问题讨论】:
假设我有一个cdata 变量(FFI 查询的结果)。然后我将其字段复制到 Lua 表中。哪个更快:访问 Lua 表或原始 cdata 变量的字段?
【问题讨论】:
让我们检查一下
使用 jit:
./tarantool -l clock -l ffi
Tarantool 2.3.1-18-ga5a2bb28a
type 'help' for interactive help
tarantool> ffi.cdef[[ typedef struct test { int field; } ]]
---
...
tarantool> T = ffi.typeof('struct test')
---
...
tarantool> F = T({12345})
---
...
tarantool> L = {field = 12345}
---
...
tarantool> local st = clock.proc() for i=1,1e8 do local x = F.field end return clock.proc() - st
---
- 0.036895
...
tarantool> local st = clock.proc() for i=1,1e8 do local x = L.field end return clock.proc() - st
---
- 0.037841
...
并且没有 jit:
tarantool> jit.off()
---
...
tarantool> local st = clock.proc() for i=1,1e6 do local x = F.field end return clock.proc() - st
---
- 0.198209
...
tarantool> local st = clock.proc() for i=1,1e6 do local x = L.field end return clock.proc() - st
---
- 0.010116
...
所以,总结一下:
【讨论】:
在禁用 JIT 的情况下,由于 LuaJIT 表 internal structure 和冲突处理优化(例如 Brent 的变体),表键查找的速度要快得多。对于cdata TGETS/TGETV 意味着元方法查找及其进一步执行。 Here 是实现字段(即键)查找的方法。
启用 JIT 后,表槽和结构字段地址/偏移量都专门用于跟踪,前面有几个断言保护。省略此类断言失败(例如由表重新分配导致的槽地址失效),性能差异可以认为可以忽略不计。
我看到 @mons-anderson 的不错的基准测试也证实了这一点。
【讨论】: