小编典典

动态JSON结构到Java结构

java

我正在一个项目中使用JSON作为创建Java对象的配置框架。这也是我来自CF / PHP / JS等方面多年经验的第一个专业Java项目。

在将JSON转换为Java时,我能找到的所有资源都基于以下想法:您必须先在Java中手动构建对象,即POJO,然后使用JSON进行填充。

作为网络语言的资深人士,我对此想法感到窒息。我发现编译后的语言起着不同的作用,但是我认为从命令行到机器语言的共享是一个原则:“如果必须执行两次以上,请使其自动化。”

……但似乎Java的主要智慧是:每次都手工构建繁琐的结构,然后使用GSON / Jackson
/任何方法填充刚性结构。如果您的JSON发生了更改(并且因为JSON而发生了更改),则将再次执行整个操作。

是否有与Web代码类似的思维方式?
1)加载JSON
2)基于JSON结构构建本地语言结构。
3)用结构中的数据填充新对象。
4)随意使用对象。

(正如@Thomas所指出的,为什么需要这样做的最准确的情况是返回JSON的API响应,它可能会定期有所不同,尽管对于高级语言而言,鸭子输入通常不舒服,甚至可以将EVERYTHING转换为字符串保存与每次都重新构建框架相比(或拥有一个将无数来源的JSON转换为可被任何其他工具使用的工具)所花费的时间和精力。同样,从牛仔语言的角度看,这似乎很明显,但对于高级语言来说却是陌生的语言人们不理解这个问题。)


阅读 330

收藏
2020-11-26

共1个答案

小编典典

我仔细阅读了上面的注释,并且认为在非常普遍的情况下,即使在编译时就知道了API结构, 临时 Java对象也将非常有用。

假设您处理一个API,该API返回一个复杂的JSON对象,您仅对非常特定的属性值感兴趣。

以以下Google Map API调用示例为例,获取地理详细信息并提供地址:

https://maps.googleapis.com/maps/api/geocode/json?address=sunnyvale,CA

如您所见,响应结构非常复杂。假设您只想获取latand lng属性值,应该有一种更简单的方法,而不是预先定义完整的对应Java类。

因此,是的,如果您不想使用强类型的预定义Java类,则需要使用原始Object类型和Maps。更具体地说,是一个StringMap。

现在终于可以给出一个解决方案:

理想情况下,您希望使用与JS中相同的本机符号来访问属性。像这样:

String url = "https://maps.googleapis.com/maps/api/geocode/json?address=" + address;
String responseStr = fetch(url);
JsonHelper response =  JsonHelper.forString(responseStr);

String status = (String) response.getValue("status");
if(status != null && status.equals("OK")) {
   lat = (Double) response.getValue("results[0].geometry.location.lat");        
   lng = (Double) response.getValue("results[0].geometry.location.lng");
}

以下 JsonHelper类代码摘自jello-framework,它使您可以准确地做到这一点。

package jello.common;

import java.util.List;

import com.google.gson.Gson;
import java.util.AbstractMap;

public class JsonHelper {

    private Object json;

    public JsonHelper(String jsonString) {
        Gson g = new Gson();
        json = g.fromJson(jsonString, Object.class);
    }

    public static JsonHelper forString(String jsonString) {
        return new JsonHelper(jsonString);
    }

    @SuppressWarnings("unchecked")
    public Object getValue(String path) {
        Object value = json;
        String [] elements = path.split("\\.");
        for(String element : elements) {
            String ename = element.split("\\[")[0];

            if(AbstractMap.class.isAssignableFrom(value.getClass())) {
                value = ( (AbstractMap<String, Object>) value).get(ename);

                if(element.contains("[")) {
                    if(List.class.isAssignableFrom(value.getClass())) {
                        Integer index = Integer.valueOf(element.substring(element.indexOf("[")+1, element.indexOf("]")) );
                        value = ((List<Object>) value).get(index);
                    }
                    else {
                        return null;
                    }
                }
            }
            else {
                return null;
            }
        }

        return value;
    }
}
2020-11-26