我正在一个项目中使用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转换为可被任何其他工具使用的工具)所花费的时间和精力。同样,从牛仔语言的角度看,这似乎很明显,但对于高级语言来说却是陌生的语言人们不理解这个问题。)
我仔细阅读了上面的注释,并且认为在非常普遍的情况下,即使在编译时就知道了API结构, 临时 Java对象也将非常有用。
假设您处理一个API,该API返回一个复杂的JSON对象,您仅对非常特定的属性值感兴趣。
以以下Google Map API调用示例为例,获取地理详细信息并提供地址:
https://maps.googleapis.com/maps/api/geocode/json?address=sunnyvale,CA
如您所见,响应结构非常复杂。假设您只想获取latand lng属性值,应该有一种更简单的方法,而不是预先定义完整的对应Java类。
lat
lng
因此,是的,如果您不想使用强类型的预定义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,它使您可以准确地做到这一点。
JsonHelper
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; } }