Lua在JS运行时(Fengari)中,大整数的字符串表示具有尾部“.0”

所以我有一个稍微复杂的问题。为了帮助解决整体问题,以下是背景:

我正在将一个应用程序转换为JavaScript,其中应用程序的一部分使用lua来解释模板。我没有控制这些模板。但是,模板解析器在lua中,它查看模板字符串是否具有以“%”开头的行或包含“\u003c%...%\u003e”标记的行。例如:

这是输出
%对于索引,值在ipairs{1,2,3}中为时,不会输出
index:<% index %>;值:<% value %>
%结束

会输出:

这是输出
index:1;value:1
index:2;value:2
index:3;value:3

不以“%”开头的行将被包装在lua帮助器“append(line_contents)”函数中,并且如果它们具有标记,例如“foo \ u003c%tag_变量% \ u003ebar”,则会被转换为“append('foo' ..(tag_variable)..'bar')”。 以“%”开头的行会变成lua代码(基本上eval'd)。

该代码运行良好,我可以控制解析模板的逻辑,并修改“append”助手。 (但不是模板代码)

因此,当模板代码与约10位数字(整数)一起使用时,主要问题会发生。在C-lua中,这些数字的字符串表示只是在模板标记中使用该值时的整数部分。但是,在Lua中JS的[Fengari runtime](https://github.com/fengari-lua/fengari)中,这些数字的某些字符串表示形式有时具有尾随“.0”。这意味着输出在运行时之间会有所不同。我正在努力找出解决此问题的方法。

我认为这是因为Fengari依赖于JS Number.prototype.toString函数以在需要时将数字格式化为字符串。看起来这是该函数的奇怪之处,会在某些时候有一些10位以上的数字开始具有尾随的小数0。_这是lua的定义行为,不是javascript,在内部表示使用浮点类型时会发生,正如Egor在评论中提到的那样。_我在Google上搜索但找不到关于这个问题的提及,但是认为它与数字的浮点表示有关。我可以使用Number.prototype.toFixed,但这并不能全局解决问题。我还想更好地了解为什么会发生这种情况,以及尝试找到解决方案。我正在考虑以某种方式覆盖Number.prototype.toString,如果这在全局上有效,那就好了,但我知道这很棘手,即使可能……此外,我可以更改模板标记的包装方式,并在输出周围添加格式化助手,这可能会在某些情况下解决问题,但这并不能帮助模板代码拼接数字的情况……那么我该如何解决这个问题呢?

参考:以下是有关如何在JS中将数字表示为字符串的规范:https://www.ecma-international.org/ecma-262/7.0/#sec-tostring-applied-to-the-number-type。但是今天我显然太累了,无法理解这种密集的数学定义 :)。

点赞
用户8031815
用户8031815

fengari的行为故意与Lua 5.3保持一致。请参见LUA_COMPAT_FLOATSTRINGhttps://www.lua.org/source/5.3/lobject.c.html#luaO_tostring和与之匹配的fengari代码:

fengari/src/lobject.js(代码0c9631c中的592至594行)

if (!LUA_COMPAT_FLOATSTRING && /^[-0123456789]+$/.test(str)) {  /* looks like an int? */
    str += '.0'; /* adds '.0' to result: lua_getlocaledecpoint removed as optimisation */
}

这在此处有所记录:

fengari/src/luaconf.js(代码0c9631c中的123至129行)

/*
@@ LUA_COMPAT_FLOATSTRING makes Lua format integral floats without a
@@ a float mark ('.0').
** This macro is not on by default even in compatibility mode,
** because this is not really an incompatibility.
*/
const LUA_COMPAT_FLOATSTRING = conf.LUA_COMPAT_FLOATSTRING || false;

可以通过使用定义LUA_COMPAT_FLOATSTRING的JSON字符串设置process.env.FENGARICONF来更改此设置。

2020-08-06 23:21:29