0

我正在使用 Poison 对 Json 进行编码和解码的伞式应用程序。为了编码我的ecto模型,我编写了以下实现:

def defimpl Poison.Encoder, for: Any do
def encode(%{__struct__: App.Models.Task} = struct, options) do
  struct =
    if struct.geom do
      {lon, lat} =  struct.geom.coordinates
      %{struct | lon: lon, lat: lat}
    else
      struct
    end
  enconde_model(struct, options)
end

def encode(%{__struct__: _} = struct, options) do
  enconde_model(struct, options)
end

def encode({lon, lat}, options) when is_float(lon) and is_float(lat) do
  %{lon: lon, lat: lat}
  |> sanitize_map()
  |> Poison.Encoder.Map.encode(options)
end

defp enconde_model(model, options) do
  model
  |> Map.from_struct()
  |> sanitize_map()
  |> Poison.Encoder.Map.encode(options)
end

defp sanitize_map(map) do
  map
  |> Enum.filter(
    fn({_, %Ecto.Association.NotLoaded{}}) -> false
      ({:__meta__, _}) -> false
      ({:__struct__, _}) -> false
      ({_, _}) -> true
    end)
  |> Map.new()
end

结尾

我的想法是,一旦我运行mix release程序就会引发错误,除非我注释掉上面显示的代码。

那么,还有另一种方法可以在不使用实现的情况下实现相同的功能吗?

4

1 回答 1

1

您收到该错误是因为您正在覆盖现有的编码器实现。正确的方法是Poison.Encoder为每个要编码的结构实现。使用@derive,这非常简单:

@derive {Poison.Encoder, only: [:the, :field, :names, :you, :want, :in, :json]}
schema "tasks" do
  ...
end

对于元组,您需要手动将它们转换为可以包含这些元组的结构的编码中的映射:

defimpl Poison.Encoder, for: Something do
  def encode(%{a_tuple: {lat, lon}, something: else_}, options) do
    Poison.encode!(%{location: %{lat: lat, lon: lon}, something: else_}, options)
  end
end
于 2018-07-04T12:37:36.027 回答