2014-11-08 57 views
0

将JPA实体转换为不包含相关对象但数据库列的JSON的最佳方式是什么?如何将JPA实体格式化为JSON和数据库列?

考虑:

public class Question implements Serializable { 

    @Id 
    @GeneratedValue(strategy = GenerationType.IDENTITY) 
    @Basic(optional = false) 
    private Integer id; 

    @Basic(optional = false) 
    @NotNull 
    @Size(min = 1, max = 255) 
    @Column(name = "question_title") 
    private String questionTitle; 

    @Size(max = 500) 
    @Column(name = "question_desc") 
    private String questionDesc; 

    @JoinColumn(name = "requires_question_id", referencedColumnName = "id") 
    @ManyToOne 
    private Question requiredQuestion; 

    ... 
} 

直接转换问题以JSON可能是这样的:

{ 
    "id": 1, 
    "questionTitle": "Q1", 
    "questionDesc": "A Question 123" 
    "requiredQuestion": { 
     "id": 2, 
     "questionTitle": "Q2", 
     "questionDesc": "A Question 456" 
    } 
} 

但是更理想的结果可能是这样的:

{ 
    "id": 1, 
    "question_title": "Q1", 
    "question_desc": "A Question 123" 
    "required_question_id": 2 
} 

是什么获得理想结果的最佳方式是什么?

创建一个像“question_title”等字符串键的hashmap并手动分配对象属性,然后将地图转换为json?

回答

1

您可以使用Jackson JSON Processor将实体序列化为JSON,就像任何其他POJO一样。要完全控制输出格式,请为Question类编写custom serializer。这将是这个样子

public class QuestionSerializer extends JsonSerializer<Question> { 
    @Override 
    public void serialize(Question value, JsonGenerator jgen, SerializerProvider provider) 
     throws IOException, JsonProcessingException { 
     jgen.writeStartObject(); 
     jgen.writeNumberField("id", value.getId()); 
     jgen.writeStringField("question_title", value.getQuestionTitle()); 
     jgen.writeStringField("question_desc", value.getQuestionDesc()); 
     jgen.writeNumberField("required_question_id", value.getRequiredQuestion().getId()); 
     jgen.writeEndObject(); 
    } 
} 
+0

何时需要任何建议问题为空?编写字符串字段时不能调用getId。简单地说,if(value.getRequiredQuestion()!= null)在它之前? – javajaba 2014-11-09 21:00:15

+1

是的,你必须检查null requiredQuestion,但是根据你想要输出的情况(根本不需要'required_question_id'字段,或者null作为值),你可以使用'jgen.writeNullField(“required_question_id”) '在'else'分支中。 – 2014-11-09 22:45:46

0

手动分配值的对象不是一个好主意,因为你可以在休眠投影让你期望的结果,然后你可以在地图转换成JSON

return sessionFactory.getCurrentSession() 
.createCriteria(Question.class) 
.createAlias("requiredQuestion","requiredQuestion") 
.setProjection(Projections.projectionList() 
    .add(Projections.property("id").as("id")) 
    .add(Projections.property("questionTitle").as("questionTitle")) 
    .add(Projections.property("questionDesc").as("questionDesc")) 
    .add(Projections.property("requiredQuestion.id").as("requiredQuestionid"))) 
    .setResultTransformer(Criteria.ALIAS_TO_ENTITY_MAP) 
.list(); 

这是得到你所期望的结果的代码,你可以将它转换为json。

1

在这种类型的应用程序中,您不应该使用相同的对象来与客户端通信并在数据库上保存数据。原因是你的表结构可能会因需求或其他原因而改变,你必须改变你的JPA实体。此外,您必须更改客户端的代码,因为JPA实体类会更改其结构。

您应该创建DTO,这是一个简单的POJO,用于向客户端传输信息(序列化)并从客户端获取信息(反序列化)。并使用像Jackson或gson这样的库将Java对象转换为Json格式和反转。