在Erlang中使用mochijson2解码JSON

时间:2010-04-29 17:19:27

标签: json erlang mochiweb mochijson2

我有一个包含一些JSON数据的var:

A = <<"{\"job\": {\"id\": \"1\"}}">>. 

使用mochijson2,我解码数据:

 Struct = mochijson2:decode(A). 

现在我有了这个:

{struct,[{<<"job">>,{struct,[{<<"id">>,<<"1">>}]}}]}

我正在尝试阅读(例如),“job”或“id”。

我尝试使用struct.get_value但它似乎不起作用。

有什么想法吗?

5 个答案:

答案 0 :(得分:13)

数据采用{struct,proplist()}格式,所以这就是你要做的:

{struct, JsonData} = Struct,
{struct, Job} = proplists:get_value(<<"job">>, JsonData),
Id = proplists:get_value(<<"id">>, Job),

您可以在http://www.erlang.org/doc/man/proplists.html

了解有关道具列表的更多信息

答案 1 :(得分:5)

访问json结构的另一个辅助函数:

jsonobj({struct,List}) ->
    fun({contains,Key}) ->
        lists:keymember(Key,1,List);
    ({raw,Key}) ->
        {_,Ret} = lists:keyfind(Key,1,List),Ret;
    (Key) ->
        {_,Ret} = lists:keyfind(Key,1,List),
        jsonobj(Ret)
    end;
jsonobj(List) when is_list(List) ->
    fun(len) ->
        length(List);
    (Index) ->
        jsonobj(lists:nth(Index,List))
    end;
jsonobj(Obj) -> Obj.

用法:

1> A=mochijson2:decode(<<"{\"job\": {\"id\": \"1\", \"ids\": [4,5,6], \"isok\": true}}">>).
2> B=jsonobj(A).
3> B(<<"job">>).
#Fun<jsonutils.1.33002110>
4> (B(<<"job">>))(<<"id">>).
1
5> (B(<<"job">>))(<<"ids">>).
#Fun<jsonutils.1.9495087>
6> (B(<<"job">>))({raw,<<"ids">>}).
[4,5,6]
7> ((B(<<"job">>))(<<"ids">>))(1).
4
8> B({raw,<<"job">>}).
{struct,[{<<"id">>,<<"1">>},
               {<<"ids">>,[1,2,3]},
               {<<"isok">>,true}]}
9> B({contains,<<"job">>}).
true
10> B({contains,<<"something">>}).
false
11> ((B(<<"job">>))(<<"ids">>))(len)
3

我认为从json中提取值不会更简单。

答案 2 :(得分:2)

这是另一种访问数据的方法。使用records syntax以方便使用。

-record(struct, {lst=[]}).

A = <<"{\"job\": {\"id\": \"1\"}}">>,
Struct = mochijson2:decode(A), 
Job = proplists:get_value(<<"job">>, Struct#struct.lst),
Id = proplists:get_value(<<"id">>, Job#struct.lst),

与使用记录的答案完全相同。使用mochijson2时的另一个选择。我个人更喜欢这种语法。

答案 3 :(得分:1)

在前面给出的答案中,mochiweb上还有一个很好的tutorial,json(视频)。

答案 4 :(得分:1)

我最喜欢处理mochijson数据的方法是用哈希映射替换所有结构,之后它们可以干净地模式匹配。为此,我写了这个易于理解的功能:

structs_to_maps({struct, Props}) when is_list(Props) ->
    lists:foldl(
        fun({Key, Val}, Map) ->
            Map#{Key => structs_to_maps(Val)}
        end,
        #{},
        Props
    );
structs_to_maps(Vals) when is_list(Vals) ->
    lists:map(
        fun(Val) ->
            structs_to_maps(Val)
        end,
        Vals
    );
structs_to_maps(Val) ->
    Val.

以下是如何使用它的示例:

do() ->
    A = <<"{\"job\": {\"id\": \"1\"}}">>,
    Data = structs_to_maps(mochijson2:decode(A)),
    #{<<"job">> := #{<<"id">> := Id}} = Data,
    Id.

这具有许多优点,尤其是在处理可能具有意外形状的传入数据时。