在业务开发中,相比于创建新的实体,我们会选择将信息保存为json字符串,作为一个字段保存。如此,每次使用时都需要显示调用json转换函数。
一个更合理的方法是,在实体类中增加一个map字段并添加@Transient注解。使用@PrePersist和@PostLoad定义保存和装载的回调函数,自动将map转换为json字符串或者将json字符串转换为map。
以下代码来自Griffin代码中的DataSource实体。需求是每次保存和更新时更新checkpoint信息,有点类似单调递增的版本控制的想法。
首先,注意两个属性,checkpoint和checkpointMap。
- checkpoint是String类型,存储json字符串,在数据库内有对应的一列,但将实体作为json输出时不会包含这一属性。
- checkpointMap属性是Map<String, Object>类型,不会被持久化到数据库,但朱提作为json输出时,如果不为null则会包括这一属性。
实现:注册两个回调函数save()和load()。在实体每次更新或者持久化到数据库之前,会先执行save()函数,将checkpointMap翻译为json字符串赋值给checkpoint属性,使得能将版本信息保存到数据库。
在实体加载完毕之后,自动调用load()函数,将checkpoint的json字符串转化为map,赋值给checkpointMap,这样我们每次使用的时候直接使用checkpointMap即可,不用再自己显式去调用json转换工具。
@JsonIgnore
@Column(length = 1024)
private String checkpoint;
@Transient
@JsonInclude(JsonInclude.Include.NON_NULL)
private Map<String, Object> checkpointMap;
@PrePersist
@PreUpdate
public void save() throws JsonProcessingException {
if (checkpointMap != null) {
// map to json string
this.checkpoint = JsonUtil.toJson(checkpointMap);
}
}
@PostLoad
public void load() throws IOException {
if (!StringUtils.isEmpty(checkpoint)) {
// json string to map
this.checkpointMap = JsonUtil.toEntity(
checkpoint, new TypeReference<Map<String, Object>>() {
});
}
}