lua循环在15个数字后卡住。

我的代码是设计用来读取数字并将它们转换成汉语拼音的:

function digitconverter (digit)
    if digit == "0" then
        cnumber = "ying2 "
    elseif digit == "1" then
        cnumber = "yi1 "
    elseif digit == "2" then
        cnumber = "er2 "
    elseif digit == "3" then
        cnumber = "san1 "
    elseif digit == "4" then
        cnumber = "si4 "
    elseif digit == "5" then
        cnumber = "wu3 "
    elseif digit == "6" then
        cnumber = "liu4 "
    elseif digit == "7" then
        cnumber = "qi1 "
    elseif digit == "8" then
        cnumber = "ba1 "
    elseif digit == "9" then
        cnumber = "jiu3 "
    end
    return cnumber
end

print("输入要转换的数字:")

repeat
    strnumber = io.read("*line")
    number = tonumber(strnumber)
    if number ~= nil then
        continue = true
    else
        print("无效输入。请重试:")
        continue = false
    end
until continue == true
nlength = #strnumber

digits = {}
for d in string.gmatch(number, "%d") do
    digits[#digits + 1] = d
end

convnumber = ""
for d=1,nlength do
    convnumber = convnumber .. digitconverter(digits[d])
end
print(convnumber)

    io.read()

如果我输入15位以上的数字,它就会卡住(说得更好听些)。它会将每个数字转换,但第16个数字会是随机的,而17及其后面的数字会重复出现另一个随机数字。我已经检查过,但找不到问题所在。有什么想法吗?

点赞
用户1847592
用户1847592

你可能想要在你的代码下面这一行中扫描 strnumber 而不是 number

for d in string.gmatch(number, "%d") do

你的 number 变量包含一个带有 15-16 位小数的 double 格式的数字值。

2013-01-30 00:26:19
用户501459
用户501459

你正在遍历 number 的数字,而不是 strnumber。问题在于当你到达太多位数时,字符串表示将会以科学计数法的形式呈现:

strnumber = '1234567890123456789'
number = tonumber(strnumber)
print(number) --> 1.2345678901235e+018

副笔记:Lua 基于哈希表,这使你(除了哈希冲突)可以实现常数时间查找。因此,你的数字转换器可以简单地编写为一个映射:

local digitmap = {
   ["0"] = "ying2 ",
   ["1"] = "yi1 ",
   ["2"] = "er2 ",
   ["3"] = "san1 ",
   ["4"] = "si4 ",
   ["5"] = "wu3 ",
   ["6"] = "liu4 ",
   ["7"] = "qi1 ",
   ["8"] = "ba1 ",
   ["9"] = "jiu3 ",
}

另外,像这样构建字符串非常低效:

for d=1,nlength do
   convnumber = convnumber .. digitconverter(digits[d])
end

你正在生成大量的中间字符串,这需要大量的分配,产生大量的垃圾。将你需要串联的所有值放入表中,然后调用 table.concat 会更快。另一个优点是你可以指定一个分隔符(当前,你将分隔符硬编码到字符串表中)。

使用这些技巧,我们可以这样重写你的代码:

local digitmap = {
   ['0'] = 'ying2',
   ['1'] = 'yi1',
   ['2'] = 'er2',
   ['3'] = 'san1',
   ['4'] = 'si4',
   ['5'] = 'wu3',
   ['6'] = 'liu4',
   ['7'] = 'qi1',
   ['8'] = 'ba1',
   ['9'] = 'jiu3',
}

print('请输入要转换的数字:')
while true do
   strnumber = io.read('*line')
   if not strnumber:match('%D') then
      break
   end
   print('无效的输入。请重新输入:')
end

local digits = {}
for digit in string.gmatch(strnumber, '%d') do
   digits[#digits + 1] = digitmap[digit]
end

print(table.concat(digits, ' '))
2013-01-30 00:33:04