1

我刚刚开始学习 LUA,我遇到了一个问题,我不确定哪种方式可以“正确”解决。当我将 Defold 传递vmath.vector3给我的函数时,它似乎是通过引用传递的,因此被更改了。
如果我将它乘以任何东西,那么这个问题就解决了。
还有另一种更正确的方法来解决这个问题吗?我不想修改作为参数传递的原始向量。

function M.get_nearest_tile(x, y)
    if y then -- if we've got 2 inputs, use x & y
        x = math.floor(x / M.TILE_SIZE)
        y = math.floor(y / M.TILE_SIZE)
        return x, y
    else -- if we've only got 1 input, use as vector
        local vec = x * 1 -- multiplying by 1 to avoid modifying the real vector
        vec.x = math.floor(vec.x / M.TILE_SIZE) 
        vec.y = math.floor(vec.y / M.TILE_SIZE) 
        return vec.x, vec.y
    end
end
4

3 回答 3

3

Defold 提供了许多在游戏开发中非常有用的特殊数据结构:

  • vector3 - vmath.vector3(x,y,z),用于描述 3D 坐标系中的位置或方向
  • vector4 - vmath.vector4(x,y,z,w),用于颜色、色调等(红色、绿色、蓝色、alpha)
  • quat - vmath.quat() 描述旋转的四元数
  • matrix4 - vmath.matrix4() 一个 4x4 值矩阵。尤其适用于视图和投影矩阵

Defold 游戏引擎使用了上述所有内容,但您也会在其他游戏引擎中找到相同类型的数据结构。

上面的数据结构有一个共同点:它们都是 Lua 类型的userdata

print(type(vmath.vector3())) -- "userdata"

用户数据始终通过引用传递,这就是您看到所描述行为的原因。Defold 确实提供了制作副本的方法:

local copy = vmath.vector3(original) -- copy the vector3 'original' local copy = vmath.vector4(original) -- copy the vector4 'original' local copy = vmath.quat(original) -- copy the quaternion 'original' local copy = vmath.matrix4(original) -- copy the matrix4 'original'

于 2018-11-14T05:17:35.270 回答
0

Since you're returning x and y as two values, you could implement both branches the same way without modifying any tables:

function M.get_nearest_tile(x, y)
    local newX, newY
    if y then -- if we've got 2 inputs, use x & y
        newX = math.floor(x / M.TILE_SIZE)
        newY = math.floor(y / M.TILE_SIZE)
    else -- if we've only got 1 input, use as vector
        newX = math.floor(x.x / M.TILE_SIZE)
        newY = math.floor(x.y / M.TILE_SIZE)
    end
    return newX, newY
end
于 2018-11-13T23:20:34.810 回答
0

您的else分支中已经有了一个解决方案:您必须先创建向量的副本,然后再对它们应用“修改”操作。

就其他选项而言,可能会想出一种使用代理表的方法,但这比仅仅创建一个副本要复杂得多。

于 2018-11-13T21:36:04.580 回答