Lua中的位操作

时间:2014-08-14 15:58:15

标签: lua bit-manipulation

我尝试解析gpx个文件并输出encoded polylinesGoogle algorithm

test.gpx

    <trkseg>
                <trkpt lon="-120.2" lat="38.5"/>
                <trkpt lon="-120.95" lat="40.7"/>
                <trkpt lon="-126.453" lat="43.252"/>
    </trkseg>

我管理了大部分内容,但编码数字时出现问题

gpx2epl:

file = io.open(arg[1], "r")
io.input(file)
--
function round(number, precision)
   return math.floor(number*math.pow(10,precision)+0.5) / math.pow(10,precision)
end

function encodeNumber(number)
  return number
end
--
local Olatitude = 0
local Olongitude = 0
--
while true do
  local line = io.read()
  if line == nil 
  then 
    break 
  end
  if string.match(line, "trkpt") then
    local latitude
    local longitude
    local encnum
    latitude = string.match(line, 'lat="(.-)"')
    longitude = string.match(line, 'lon="(.-)"')
    latitude = round(latitude,5)*100000
    longitude = round(longitude,5)*100000
    encnum = encodeNumber(latitude-Olatitude)
    print(encnum)
    encnum = encodeNumber(longitude-Olongitude)
    print(encnum)
    Olatitude = latitude
    Olongitude = longitude
  end
end

此脚本会生成预期输出(请参阅:Google Link),编码纬度和经度除外。

3850000
-12020000
220000
-75000
255200
-550300

Mapquest在Javascript中提供了一个实现:

function encodeNumber(num) {
   var num = num << 1;
   if (num < 0) {
      num = ~(num);
   }
   var encoded = '';
   while (num >= 0x20) {
      encoded += String.fromCharCode((0x20 | (num & 0x1f)) + 63);
      num >>= 5;
   }
   encoded += String.fromCharCode(num + 63);
   return encoded;   
}

这可以在Lua完成吗?有人可以帮帮我。我不知道如何在Lua中实现它。

修改

根据Doug的建议,我做了:

function encodeNumber(number)
  local num = number
  num = num * 2
  if num < 0
  then
    num = (num * -1) - 1
  end
  while num >= 32
  do
    local num2 = 32 + (num % 32) + 63
    print(string.char(num2))
    num = num / 32
  end
  print(string.char(num + 63) .. "\n-----")
end

encodeNumber(3850000)   -- _p~iF
encodeNumber(-12020000) -- ~ps|U
encodeNumber(220000)    -- _ulL
encodeNumber(-75000)    -- nnqC
encodeNumber(255200)    -- _mqN
encodeNumber(-550300)   -- vxq`@

接近预期的输出,但只接近...任何提示?

1 个答案:

答案 0 :(得分:3)

encodeNumber零碎......

   var num = num << 1;

这只是num = num * 2

   num = ~(num);

这是num = (- num) - 1

   0x20 | (num & 0x1f)

相当于32 + (num % 32)

   num >>= 5

相当于num = math.floor(num / 32)

附录

要连接字符,请使用表格收集它们:

function encodeNumber(number)
  local num = number
  num = num * 2
  if num < 0
  then
    num = (num * -1) - 1
  end
  local t = {}
  while num >= 32
  do
    local num2 = 32 + (num % 32) + 63
    table.insert(t,string.char(num2))
    num = math.floor(num / 32) -- use floor to keep integer portion only
  end
  table.insert(t,string.char(num + 63))
  return table.concat(t)
end