有可靠的方法可以可靠地区分 cdata 和 ctypes吗?
2021-3-31 7:44:2
收藏:0
阅读:220
评论:2
我是一个需要区别 cdata 和 ctypes 的库的作者。目前,我使用 ffi.typeof(值)== 值,但对于一些定义了__eq的metatable的cdata会出错。提出此错误报告的用户建议检查 tostring(value):match“^ ctype”,但这只是将问题移动到定义了__tostring的cdata。我尝试过rawequal,但我的原始条件依赖于元魔术,所以根本不工作。
我正在考虑使用 local success,eq = pcall(function()return ffi.typeof(value)== value end) 并使用条件 success and eq,但我想知道:是否有更可靠,不太 hack-y 的方法来实现这一点?
点赞
用户5688146
这个问题在一段时间前也困扰着我,下面是我所了解到的情况。
这份代码对我的用例可以正常工作,但是它无法处理这种边缘情况:
assert(ffi.new("int", 9) == ffi.typeof("int")) --> 这个断言是正确的。
我找不到解决方法,我猜这就是 LuaJIT 的工作方式。对于其他基本类型(和它们各自的 id),情况也是一样的。
完整代码:
local ffi = require "ffi"
function isctype(o)
-- ctypes 可以转化为数字
local check = tonumber(o)
if check == nil then
return false
end
-- typeinfo 需要一个数字,我们有一个数字
local maybeInfo = ffi.typeinfo(check)
local sureInfo = ffi.typeinfo(ffi.typeof(o))
-- 类型信息不同
if maybeInfo.info ~= sureInfo.info then
return false
end
local type = ffi.typeof(o)
-- 除了调用 tostring(o) ,没有其他方法可以避免这个问题。
-- 这会有它自己的缺点。
-- 当我们到达这里时,我们有一个特殊的情况
-- 类型 id 等于 tonumber(o) 的结果。
-- 这种情况发生在 typeof"int" 和 9 的情况下。
-- 在这种情况下,结果也是 true,
-- 这实际上并不完全符合我的期望。
return o == type
end
local ct = ffi.typeof("struct { int x; }")
print("type", isctype(ct)) --> true
print("data", isctype(ffi.new(ct))) --> false
print("type", isctype(ffi.typeof("int"))) --> true
print("type", isctype(ffi.typeof("const char*"))) --> true
print("data", isctype(ffi.new("float", 3))) --> false
print("data", isctype(ffi.new("int", 3))) --> false
print("data", isctype(ffi.new("intptr_t", 3))) --> false
print("data", isctype(ffi.new("int[1]", 3))) --> false
print("data", isctype(ffi.new("const char*", "yo"))) --> false
print("data", isctype(ffi.new("int", 8))) --> false
-- 这很令人失望 :(
print("data", isctype(ffi.new("int", 9))) --> true
assert(ffi.new("int", 9) == ffi.typeof("int"))
print("assertion did not fail :(")
2021-04-14 20:43:56
评论区的留言会收到邮件通知哦~
推荐文章
- Lua 虚拟机加密load(string.dump(function)) 后执行失败问题如何解决
- 我想创建一个 Nginx 规则,禁止访问
- 如何将两个不同的lua文件合成一个 东西有点长 大佬请耐心看完 我是小白研究几天了都没搞定
- 如何在roblox studio中1:1导入真实世界的地形?
- 求解,lua_resume的第二次调用继续执行协程问题。
- 【上海普陀区】内向猫网络招募【Skynet游戏框架Lua后端程序员】
- SF爱好求教:如何用lua实现游戏内调用数据库函数实现账号密码注册?
- Lua实现网站后台开发
- LUA错误显式返回,社区常见的规约是怎么样的
- lua5.3下载库失败
- 请问如何实现文本框内容和某个网页搜索框内容连接,并把网页输出来的结果反馈到另外一个文本框上
- lua lanes多线程使用
- 一个kv数据库
- openresty 有没有比较轻量的 docker 镜像
- 想问一下,有大佬用过luacurl吗
- 在Lua执行过程中使用Load函数出现问题
- 为什么 neovim 里没有显示一些特殊字符?
- Lua比较两个表的值(不考虑键的顺序)
- 有个lua简单的项目,外包,有意者加微信 liuheng600456详谈,最好在成都
- 如何在 Visual Studio 2022 中运行 Lua 代码?

下面的解决方案也比较hack-y,但可能对您有所帮助。
它滥用了未记录的特性,即
tonumber()可适用于ctypes。local function is_ctype(x) return type(x) == "cdata" and tonumber(x) ~= nil and tostring(x):sub(1,5) == "ctype" end请注意,
tostring()仅适用于可转换为数字的cdata。ctype<complex>是唯一的可元类型ctype,其值可转换为数字。令人高兴的是,
ctype<complex>不尊重__tostring元方法。 :-)