"interval is empty",Lua的math.random无法处理大数字?
2013-11-24 5:51:14
收藏:0
阅读:240
评论:2
我不知道这是Lua自身的错误还是我做错了什么。我在使用Lua for Windows (Lua 5.1.4):
>return math.random(0, 1000000000)
1251258
这会返回0到10000000000之间的随机整数,与预期一致。这似乎对所有其他值都有效。但是,如果我加一个0:
>return math.random(0, 10000000000)
stdin:1: bad argument #2 to 'random' (interval is empty)
任何大于那个值的数字都会出现同样的问题。
我试图找出到底需要多高的数字才会导致这种情况,发现了更奇怪的东西:
>return math.random(0, 2147483647)
-75617745
如果这个值是2147483647,那么它会给我负数。任何比这更高的值都会报错。任何比这更低的数字都可以正常工作。
这是二进制中的0b1111111111111111111111111111111,确切地说是31个二进制数字。不过我不确定这意味着什么。
点赞
用户869951
如果你需要比随机函数支持的范围(32位有符号整数或2^31,因为math.random在C级别)更大但小于Lua“number”类型的范围(基于What is the maximum value of a number in Lua?,2^52,甚至可能是2^53),你可以尝试生成两个随机数:将第一个缩放到所需范围;使用第二个“填充间隙”。例如,假设你想要0到2^36的范围。从math.random得到的最大值是2^31。因此,你可以这样做:
-- 2^36 = 2^31 * 2^5 这样
scale = 2^5
baseRand = scale * math.random(0, 2^31)
-- baseRand 现在在 0 到 2^36 的范围内但是在可能的值集合中存在 2^5 的间隙; 用第二个随机数填充间隙:
fillGap = math.random(0, 2^5)
randNum = baseRand + fillGap
只要所需范围小于Lua解释器对Lua数字的最大值,这将起作用,该值是可配置的编译时参数,但如果你使用股票构建,则为2^52,一个非常大的数字(虽然不像最大长整数2^63那样大)。
还要注意最大正N位整数为2^N-1(不是2^N),但上述技术可应用于任何范围,例如scale = 10^6 ,则randNum = 10^6 * math.random(0, 10^8) + math.random(0, 10^6)。
2013-11-24 14:46:15
评论区的留言会收到邮件通知哦~
推荐文章
- 如何将两个不同的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 代码?
- addEventListener 返回 nil Lua
- Lua中获取用户配置主目录的跨平台方法
这个意外的行为(bug?) 是由于 Lua 5.1 中
math.random如何处理输入参数导致的。从lmathlib.c中可以看到:case 2: { /* 底部和上限 */ int l = luaL_checkint(L, 1); int u = luaL_checkint(L, 2); luaL_argcheck(L, l<=u, 2, "interval is empty"); lua_pushnumber(L, floor(r*(u-l+1))+l); /* 在 `l' 和 `u' 之间的 int */ break; }你可能知道,在 C 中,标准的
int可以表示-2,147,483,648到2,147,483,647的值。像你的用例中那样将+1添加到2,147,483,647会导致溢出并将值 回绕 给-2,147,483,648。最终结果是负数,因为你正数与负数相乘。此外,超过
2,147,483,647的任何数都会由于溢出回绕而导致luaL_argcheck失败。有几种方法可以解决这个问题:
lua_Number来解决此问题。