我使用com.google.gson.Gson
API将JSON字符串转换为Java对象:
Gson gson = new Gson();
User u = gson.fromJson(jsonString, User.class);
我想知道 javax.json api中是否有等效的API来执行等效操作。 谢谢!
答案 0 :(得分:8)
对于像Gson这样的oneliner来说,这绝对是不可能的。 javax.json
API的级别非常低。它只返回一个JSON对象结构,您可以自行分解并映射到javabean。
为了达到(几乎)与Gson#fromJson()
相同的效果,这里是样板代码的启动示例。在某种程度上支持参数化类型和javabeans。它是所有标准的Java SE API,javabeans在java.beans
API的帮助下进行了内省。
@SuppressWarnings("unchecked")
public static <T> T fromJson(String json, Class<T> beanClass) {
JsonValue value = Json.createReader(new StringReader(json)).read();
return (T) decode(value, beanClass);
}
private static Object decode(JsonValue jsonValue, Type targetType) {
if (jsonValue.getValueType() == ValueType.NULL) {
return null;
}
else if (jsonValue.getValueType() == ValueType.TRUE || jsonValue.getValueType() == ValueType.FALSE) {
return decodeBoolean(jsonValue, targetType);
}
else if (jsonValue instanceof JsonNumber) {
return decodeNumber((JsonNumber) jsonValue, targetType);
}
else if (jsonValue instanceof JsonString) {
return decodeString((JsonString) jsonValue, targetType);
}
else if (jsonValue instanceof JsonArray) {
return decodeArray((JsonArray) jsonValue, targetType);
}
else if (jsonValue instanceof JsonObject) {
return decodeObject((JsonObject) jsonValue, targetType);
}
else {
throw new UnsupportedOperationException("Unsupported json value: " + jsonValue);
}
}
private static Object decodeBoolean(JsonValue jsonValue, Type targetType) {
if (targetType == boolean.class || targetType == Boolean.class) {
return Boolean.valueOf(jsonValue.toString());
}
else {
throw new UnsupportedOperationException("Unsupported boolean type: " + targetType);
}
}
private static Object decodeNumber(JsonNumber jsonNumber, Type targetType) {
if (targetType == int.class || targetType == Integer.class) {
return jsonNumber.intValue();
}
else if (targetType == long.class || targetType == Long.class) {
return jsonNumber.longValue();
}
else {
throw new UnsupportedOperationException("Unsupported number type: " + targetType);
}
}
private static Object decodeString(JsonString jsonString, Type targetType) {
if (targetType == String.class) {
return jsonString.getString();
}
else if (targetType == Date.class) {
try {
return new SimpleDateFormat("MMM dd, yyyy H:mm:ss a", Locale.ENGLISH).parse(jsonString.getString()); // This is default Gson format. Alter if necessary.
}
catch (ParseException e) {
throw new UnsupportedOperationException("Unsupported date format: " + jsonString.getString());
}
}
else {
throw new UnsupportedOperationException("Unsupported string type: " + targetType);
}
}
private static Object decodeArray(JsonArray jsonArray, Type targetType) {
Class<?> targetClass = (Class<?>) ((targetType instanceof ParameterizedType) ? ((ParameterizedType) targetType).getRawType() : targetType);
if (List.class.isAssignableFrom(targetClass)) {
Class<?> elementClass = (Class<?>) ((ParameterizedType) targetType).getActualTypeArguments()[0];
List<Object> list = new ArrayList<>();
for (JsonValue item : jsonArray) {
list.add(decode(item, elementClass));
}
return list;
}
else if (targetClass.isArray()) {
Class<?> elementClass = targetClass.getComponentType();
Object array = Array.newInstance(elementClass, jsonArray.size());
for (int i = 0; i < jsonArray.size(); i++) {
Array.set(array, i, decode(jsonArray.get(i), elementClass));
}
return array;
}
else {
throw new UnsupportedOperationException("Unsupported array type: " + targetClass);
}
}
private static Object decodeObject(JsonObject object, Type targetType) {
Class<?> targetClass = (Class<?>) ((targetType instanceof ParameterizedType) ? ((ParameterizedType) targetType).getRawType() : targetType);
if (Map.class.isAssignableFrom(targetClass)) {
Class<?> valueClass = (Class<?>) ((ParameterizedType) targetType).getActualTypeArguments()[1];
Map<String, Object> map = new LinkedHashMap<>();
for (Entry<String, JsonValue> entry : object.entrySet()) {
map.put(entry.getKey(), decode(entry.getValue(), valueClass));
}
return map;
}
else try {
Object bean = targetClass.newInstance();
for (PropertyDescriptor property : Introspector.getBeanInfo(targetClass).getPropertyDescriptors()) {
if (property.getWriteMethod() != null && object.containsKey(property.getName())) {
property.getWriteMethod().invoke(bean, decode(object.get(property.getName()), property.getWriteMethod().getGenericParameterTypes()[0]));
}
}
return bean;
}
catch (Exception e) {
throw new UnsupportedOperationException("Unsupported object type: " + targetClass, e);
}
}
用法:
User u = YourJsonUtil.fromJson(jsonString, User.class);
如果您发现来自UnsupportedOperationException
方法之一的decodeXxx()
,只需将所需的转化逻辑添加到相关方法即可。当然,这可以进一步重构以应用策略模式,以便它具有灵活性和可扩展性,但是我们基本上重新发明了Gson。
答案 1 :(得分:2)
不,没有。
在你尝试其他解析器之前,然后GSON:org.json有错误/有实现错误。 QuickJson有错误/有实施错误