如何将 Erlang 对象结构转换为 Elixir Map?
How to convert Erlang object structure to Elixir Map?
我正在使用 couchbeam 从 Elixir 联系 CouchDB。
但是 lib 返回旧的 erlang 对象表示形式,如 {[{"foo", "bar"}]}
而不是 elixir 映射,这是由于 lib 使用 jiffy:decode 而没有 return_maps
,我该如何转换它对象结构到 Elixir 映射(反之亦然)?
我找到了一个 jiffy:encode 和 jiffy:decode 的 hackish 方法 return_maps
...但是必须有另一种选择吗?
更新:
来自 Hynek 在 erlang 中的示例,这似乎有效:
defmodule ToMaps do
def convert({x}) when is_list(x) do
Map.new(x, fn {k, v} -> {k, convert(v)} end)
end
def convert([head | tail]) do
[convert(head) | convert(tail)]
end
def convert(x) do
x
end
end
似乎可以胜任。
iex(1)> ToMaps.convert({[{"foo",[{[{"a",1}]},3]},{"bar","baz"}]})
%{"bar" => "baz", "foo" => [%{"a" => 1}, 3]}
我不知道 Elixir,但在 Erlang 中:
-module(to_maps).
-export([to_maps/1]).
to_maps({L}) when is_list(L) ->
maps:from_list([{K, to_maps(V)} || {K, V} <- L]);
to_maps([H|T]) ->
[to_maps(H) | to_maps(T)];
to_maps(X) -> X.
编辑:
反转:
from_maps(#{} = M) ->
F = fun(K, V, Acc) -> [{K, from_maps(V)} | Acc] end,
{maps:fold(F, [], M)};
from_maps([H|T]) ->
[from_maps(H) | from_maps(T)];
from_maps(X) -> X.
我不推荐它,但它甚至可以是一个功能:
convert({L}) when is_list(L) ->
maps:from_list([{K, convert(V)} || {K, V} <- L]);
convert(#{} = M) ->
F = fun(K, V, Acc) -> [{K, convert(V)} | Acc] end,
{maps:fold(F, [], M)};
convert([H|T]) ->
[convert(H) | convert(T)];
convert(X) -> X.
用法:
1> jiffy:decode(<<"{\"foo\":[3, {\"a\":1}], \"bar\":\"baz\"}">>).
{[{<<"foo">>,[3,{[{<<"a">>,1}]}]},{<<"bar">>,<<"baz">>}]}
2> to_maps:to_maps(v(-1)).
#{<<"bar">> => <<"baz">>,<<"foo">> => [3,#{<<"a">> => 1}]}
3> to_maps:from_maps(v(-1)).
{[{<<"foo">>,[3,{[{<<"a">>,1}]}]},{<<"bar">>,<<"baz">>}]}
4> to_maps:convert(v(-1)).
#{<<"bar">> => <<"baz">>,<<"foo">> => [3,#{<<"a">> => 1}]}
5> to_maps:convert(v(-1)).
{[{<<"foo">>,[3,{[{<<"a">>,1}]}]},{<<"bar">>,<<"baz">>}]}
6> to_maps:convert(v(-1)).
#{<<"bar">> => <<"baz">>,<<"foo">> => [3,#{<<"a">> => 1}]}
...
我需要查看你所有的数据结构才能确定,但你可以使用这样的东西:
iex(1)> Enum.into([{"foo", "bar"}], %{})
%{"foo" => "bar"}
我正在使用 couchbeam 从 Elixir 联系 CouchDB。
但是 lib 返回旧的 erlang 对象表示形式,如 {[{"foo", "bar"}]}
而不是 elixir 映射,这是由于 lib 使用 jiffy:decode 而没有 return_maps
,我该如何转换它对象结构到 Elixir 映射(反之亦然)?
我找到了一个 jiffy:encode 和 jiffy:decode 的 hackish 方法 return_maps
...但是必须有另一种选择吗?
更新:
来自 Hynek 在 erlang 中的示例,这似乎有效:
defmodule ToMaps do
def convert({x}) when is_list(x) do
Map.new(x, fn {k, v} -> {k, convert(v)} end)
end
def convert([head | tail]) do
[convert(head) | convert(tail)]
end
def convert(x) do
x
end
end
似乎可以胜任。
iex(1)> ToMaps.convert({[{"foo",[{[{"a",1}]},3]},{"bar","baz"}]})
%{"bar" => "baz", "foo" => [%{"a" => 1}, 3]}
我不知道 Elixir,但在 Erlang 中:
-module(to_maps).
-export([to_maps/1]).
to_maps({L}) when is_list(L) ->
maps:from_list([{K, to_maps(V)} || {K, V} <- L]);
to_maps([H|T]) ->
[to_maps(H) | to_maps(T)];
to_maps(X) -> X.
编辑:
反转:
from_maps(#{} = M) ->
F = fun(K, V, Acc) -> [{K, from_maps(V)} | Acc] end,
{maps:fold(F, [], M)};
from_maps([H|T]) ->
[from_maps(H) | from_maps(T)];
from_maps(X) -> X.
我不推荐它,但它甚至可以是一个功能:
convert({L}) when is_list(L) ->
maps:from_list([{K, convert(V)} || {K, V} <- L]);
convert(#{} = M) ->
F = fun(K, V, Acc) -> [{K, convert(V)} | Acc] end,
{maps:fold(F, [], M)};
convert([H|T]) ->
[convert(H) | convert(T)];
convert(X) -> X.
用法:
1> jiffy:decode(<<"{\"foo\":[3, {\"a\":1}], \"bar\":\"baz\"}">>).
{[{<<"foo">>,[3,{[{<<"a">>,1}]}]},{<<"bar">>,<<"baz">>}]}
2> to_maps:to_maps(v(-1)).
#{<<"bar">> => <<"baz">>,<<"foo">> => [3,#{<<"a">> => 1}]}
3> to_maps:from_maps(v(-1)).
{[{<<"foo">>,[3,{[{<<"a">>,1}]}]},{<<"bar">>,<<"baz">>}]}
4> to_maps:convert(v(-1)).
#{<<"bar">> => <<"baz">>,<<"foo">> => [3,#{<<"a">> => 1}]}
5> to_maps:convert(v(-1)).
{[{<<"foo">>,[3,{[{<<"a">>,1}]}]},{<<"bar">>,<<"baz">>}]}
6> to_maps:convert(v(-1)).
#{<<"bar">> => <<"baz">>,<<"foo">> => [3,#{<<"a">> => 1}]}
...
我需要查看你所有的数据结构才能确定,但你可以使用这样的东西:
iex(1)> Enum.into([{"foo", "bar"}], %{})
%{"foo" => "bar"}