1、背景:
目前的我们项目大量的请求模型DTO和返回模型DTO。
场景1:当需要修改一条记录的时候,前端传入后端的参数需要使用请求DTO接收,然后根据id找到对应的持久化层对象,当请求模型DTO的属性和持久化层模型属性不一致的时候,属性之间赋值需要使用对象属性set的方式进行赋值,或者是其他的框架(如MapStruts等),总之非常不方便。
场景2:在进行查询的时候,需要将查询到的持久化层对象通过对象set的方式(或MapStruts框架)转换为自定义的返回模型DTO。
如上两种方式都是费时费力且代码不优雅、可维护性、可读性较差。
2、解决方案:
直接使用JPA数据库模型对象作为接收(返回)对象,实现基本的增、删、改、查的功能,完全不需要定义请求模型DTO和返回模型DTO,以及对象属性之间的赋值操作。
3、技术实现:
3.1、持久化层模型
3.1.1、一的一方的模型
@Table(name = "one")
public class One {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Integer id;
@OneToMany(cascade = CascadeType.ALL, fetch = FetchType.LAZY, mappedBy = "one")
List<Many> many;
}
3.1.2、多的一方的模型
@ToString(exclude = "one")
@JSONType(ignores = "one")
public class Many {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Integer id;
@JsonIgnore
@ManyToOne
private One one;
}
3.2:重点说明:
3.2.1、必选注解:
多的一方关联的一的一方的属性上添加:@JsonIgnore,解决json循环引用栈内存溢出的问题:
org.springframework.http.converter.HttpMessageNotWritableException: Could not write JSON: Infinite recursion (StackOverflowError);
3.2.2、非必须注解:
不对!!!如果将持久化层对象转换为com.alibaba.fastjson.JSONObject需要添加如下注解:多的一方的类添加:@JSONType(ignores = "one"),解决返回的多的一方对象中存在"$ref"属性的问题
正确的解决思路:
//SerializerFeature.WriteMapNullValue:解决null不序列化的问题
//SerializerFeature.DisableCircularReferenceDetect:消除对同一对象循环引用的问题,默认为false。当进行toJSONString的时候,默认如果重用对象的话,会使用引用的方式进行引用对象。
template.convertAndSend(exchange, null, JSONObject.toJSONString(message, SerializerFeature.WriteMapNullValue, SerializerFeature.DisableCircularReferenceDetect));
3.2.3、非必须注解:
在多的一方的类上添加:@ToString(exclude = "one"),解决在断点调试的时候不出现栈溢出问题,可以添加如上注解重写toString方法的时候排除one属性:
Method threw 'java.lang.StackOverflowError' exception. Cannot evaluate com.efivestar.xdsh.ui.domain.knowledge.KnowledgeCategory.toString()