- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我正在尝试使用 Jackson 在 JSON 文件和具有两个 Java 子类的抽象类之间进行转换。理想情况下,我想使用如下 JSON:
没有封装的Json文档
[ {
"type" : "lion",
"name" : "Simba",
"endangered" : true,
"action" : "running"
}, {
"type" : "elephant",
"name" : "Dumbo",
"endangered" : false,
"table" : [ 1.0, 2.0, 3.0 ]
} ]
我已经注释了 Animal
抽象类,如 http://www.studytrails.com/java/json/java-jackson-Serialization-polymorphism.jsp 所示。
@JsonTypeInfo(use = JsonTypeInfo.Id.NAME, include = As.PROPERTY, property = "type")
@JsonSubTypes({ @Type(value = Lion.class, name = "lion"),
@Type(value = Elephant.class, name = "elephant") })
public abstract class Animal {
String name;
boolean endangered;
//Skipped constructor/getters/setters
}
我可以使用包装器对象成功读/写它,但生成的 JSON 文件将包含一个额外的 animals
对象。
带包装器的 JSON 文档
{
"animals" : [ {
"type" : "lion",
"name" : "Simba",
"endangered" : true,
"action" : "running"
}, {
"type" : "elephant",
"name" : "Dumbo",
"endangered" : false,
"table" : [ 1.0, 2.0, 3.0 ]
} ]
}
当我使用普通 Java 列表时,我可以在没有包装器的情况下成功地从 Json 文档中读取,但是如果我尝试编写它,输出文件将如下所示:
输出.json
[ {
"name" : "Simba",
"endangered" : true,
"action" : "running"
}, {
"name" : "Dumbo",
"endangered" : false,
"table" : [ 1.0, 2.0, 3.0 ]
} ]
Java代码
// Test reading using raw list
JavaType listType = mapper.getTypeFactory()
.constructCollectionType(List.class, Animal.class);
List<Animal> jsonList = mapper.readValue(new FileInputStream(
"demo.json"), listType);
jsonDocument = new File(outputFile);
// Test writing using raw list
mapper.writerWithDefaultPrettyPrinter().writeValue(jsonDocument,
jsonList);
有什么想法可以在将对象序列化为 JSON 时包含类型信息吗?
可以在以下位置找到完整的 Eclipse 项目: https://github.com/nkatsar/json-subclass
编辑:简化了 Java 代码 以使用 JavaType
而不是 TypeReference
。GitHub 上的代码也更新了正确的解决方案
编辑 2:如评论中所述,我最终使用对象数组进行序列化/反序列化,如下所示:
// Test reading using array
Animal[] jsonArray = mapper.readValue(
new FileInputStream(demoFile), Animal[].class);
System.out.println("Reading using array:\nObject: "
+ Arrays.toString(jsonArray));
// Test writing using array
outputJson = mapper.writeValueAsString(jsonArray);
System.out.println("Writing using array:\nJSON: " + outputJson);
最佳答案
这里的问题是Java类型删除,也就是说List对象的标称类型是List。并且由于 java.lang.Object 没有@JsonTypeInfo,它不会被启用。
但您可以配置 Jackson 使用特定的 Java 类型来转换 List 元素:
JavaType listJavaType = mapper.getTypeFactory().constructCollectionType(List.class, Animal.class);
String outputJson = mapper.writerWithType(listJavaType).writeValueAsString(myList);
修改了你的代码:
public static void main(String[] args) {
List<Animal> myList = new ArrayList<Animal>();
myList.add(new Lion("Simba"));
// myList.add(new Lion("Nala"));
myList.add(new Elephant("Dumbo"));
// myList.add(new Elephant("Lucy"));
AnimalList wrapperWrite = new AnimalList(myList);
try {
ObjectMapper mapper = new ObjectMapper();
JavaType listJavaType = mapper.getTypeFactory().constructCollectionType(List.class, Animal.class);
String outputJson = mapper.writerWithType(listJavaType).writeValueAsString(myList);
System.out.println(outputJson);
List wrapperRead = mapper.readValue(outputJson, List.class);
System.out.println("Read recently generated data: " + wrapperRead);
// Test reading using raw list
List<Animal> jsonList = mapper.readValue(new FileInputStream( "demo.json"), listJavaType);
System.out.println("Read from demo.json \ndata: " + jsonList);
outputJson = mapper.writerWithType(listJavaType).writeValueAsString(jsonList);
System.out.println(outputJson);
} catch (JsonGenerationException e) {
System.out.println("Could not generate JSON: " + e.getMessage());
} catch (JsonMappingException e) {
System.out.println("Invalid JSON Mapping: " + e.getMessage());
} catch (IOException e) {
System.out.println("File I/O error: ");
}
结果:
[{"type":"lion","name":"Simba","endangered":false,"action":null},{"type":"elephant","name":"Dumbo","endangered":false,"table":null}]
Read recently generated data: [{type=lion, name=Simba, endangered=false, action=null}, {type=elephant, name=Dumbo, endangered=false, table=null}]
Read from demo.json
data: [Animal(name=Nala, endangered=true, action=running}, Animal(name=Lucy, endangered=false, table=[1.0, 2.0, 3.0]}]
[{"type":"lion","name":"Nala","endangered":true,"action":"running"},{"type":"elephant","name":"Lucy","endangered":false,"table":[1.0,2.0,3.0]}]
另一个解决方案:您可以使用自定义 TypeReference 来通知 Jackson 需要使用什么类来转换 List:
mapper.writerWithType(new TypeReference<List<Animal>>() {}).writeValueAsString(myList)
该解决方案也可以工作,但我更喜欢主解决方案,因为它更干净,您不需要实例化抽象的“TypeReference”类...
关于java - 在不使用包装类的情况下使用 Jackson 序列化 java 对象时维护子类型信息,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27426327/
我们已经在我的工作场所使用 SVN 几年了,自从我们安装它以来,除了更新和备份之外,我们真的没有对其进行任何类型的维护。我们还应该做些什么来维护 SVN,或者我们已经做了所有我们真正需要做的事情吗?
正则表达式模式如下: ".*/.*/.*/.*/.*/.*/(.*)-\d{2}\.\d{2}\.\d{2}.\d{4}.*" 确实很难维护。 我想知道,有没有这样的东西: ".*/.*/.*/.*/
我已经搜索了一些,但没有找到任何对我有帮助的问题/答案。问题是我的 jQuery 函数调用变得太大而无法维护。我想知道我是否应该进行更多重构,或者是否有更好的方法来完成所有这些调用。当我进行一次调用时
我在 mySql 中有一个记录表。我需要按照用户指定的方式为它们维护订单。所以我添加了一个“位置”列。 当我移动特定记录时更新所有记录的 SQL 语句是什么?我有类似的东西: UPDATE items
我正在使用 go channels 作为类似队列的机制,这非常适合我。我正在为每个用户打开这些类似队列的 channel 之一,并为这些 channel 中的每一个都有一个 for-range 循环。
使用 docker,您可以非常好地基于其他图像创建图像。例如,您可以制作一个镜像 Java-jdk7(基于最新的 Ubuntu LTS),并在此基础上创建镜像 elastic-search 和 tom
我正在用 Bash 编写脚本。 我的关联数组有问题,当我像这样在我的数组中放置一条记录时: declare -A arr_list_people_name 我将文本放入循环关联数组的方式(将文本排序)
我目前正在开发一个系统,该系统需要在没有可用互联网连接的情况下安装 python(或者至少我不能假设有可用的互联网连接), 我想知道维护 PIP 存储库的间接费用是多少,而且这样的存储库也可能会满足系
我正在考虑使用 Chrome 扩展的国际化支持,如 here 所述. 建议的翻译方法是先创建英文 messages.json 文件,然后将其复制并翻译成给定的语言。 我的问题是,这对于初始翻译来说工作
我想在(自托管)bitbucket 服务器中克隆 github 存储库,并不时从 github 存储库中提取最新更改。在我们的克隆中,我们将做一些永远不会离开我们的存储库的实验性内容。 为了显示;对于
我的应用程序基于银行域,需要 session 处理。当应用程序空闲时(应用程序打开后没有任何触摸事件)必须在后台计算时间。 当应用程序进入前台时,我处理 session 维护以及 AppDelegat
我可以保持 UISegmentViewControl 段的选定状态吗?即,即使用户选择了另一个段,也可以保持一个段显示为选中状态?我似乎在任何地方都找不到任何可以做到这一点的东西!! 最佳答案 这是不
我的要求:我想将登录详细信息(电子邮件、密码)发送到服务器,必须保持有效用户名的 session 。 如何使用 iphone SDK 的“NSURLConnection”创建和维护 session ?
就像Carl's question over here我想问你(因为我自己找不到 :( ) 删除既不是静态也不是动态(例如通过反射)使用的程序集引用是否有任何好处。 最佳答案 除了清理项目之外,删除未
我使用的是Bootstrap 3。我目前有2个页面,一个是查看页面,一个是编辑页面。两个页面都有许多导航选项卡,例如 id= tab1、tab2、tab3。 我想要实现的是,当我在查看页面的 tab2
我正在创建 Chrome 应用程序,我希望我的用户在首次进入应用程序时登录或创建用户。 目标: 在 Chrome 打包的应用程序上维护登录状态。 问题: Cookie - Chrome 打包的应用程序
我有arm模板来使用资源及其设置重新创建资源组。这工作得很好。 用例: 一些开发人员访问 Azure 门户并更新某些资源的某些设置。有没有办法获得可以应用于我的模板的精确更改以使这些更改生效? (更新
我有一个包含三个组合框的表单,一个代表该月(可能的)31 天,第二个代表代表月份的 12 个数字,第三个代表与 future 五年相对应的年份值。 我将它们连接在一起形成一个日期 TheDay = C
我有一个打开多个 JIF 的应用程序,但我只想创建 JIF 的单个实例,因此我使用这些函数来检查这一点,并在按下某个键后使用 dispose 关闭 JIF(JDesktopPane. getSelec
我想为一个项目制作一个帐户屏幕,但我对 GUI 还很陌生。这是我第一次使用 JComboBox,但遇到了一些麻烦。我基本上想将 JComboBox 放置在一个盒子内,这将成为我的背景图像的一部分。我尝
我是一名优秀的程序员,十分优秀!