直接使用JPA数据库模型对象作为接收(返回)对象,实现基本的增、删、改、查的功能

13 篇文章 0 订阅

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()

JPA 中,你可以使用以下几种方式来返回自定义对象: 1. 使用构造函数表达式:你可以在询中使用构造函数表达式来创建自定义对象。假设你有一个名为 `CustomObject` 的类,它有一个带有参数的构造函数,你可以在询中使用这个构造函数来返回自定义对象。例如: ```java String jpql = "SELECT NEW com.example.CustomObject(c.name, c.age) FROM Customer c"; TypedQuery<CustomObject> query = entityManager.createQuery(jpql, CustomObject.class); List<CustomObject> resultList = query.getResultList(); ``` 2. 使用 `@SqlResultSetMapping`:你可以在实体类中使用 `@SqlResultSetMapping` 注解来定义结果映射。首先,在实体类中定义一个构造函数,然后在 `@SqlResultSetMapping` 注解中指定该构造函数的参数映射。例如: ```java @SqlResultSetMapping( name = "CustomObjectMapping", classes = @ConstructorResult( targetClass = CustomObject.class, columns = { @ColumnResult(name = "name", type = String.class), @ColumnResult(name = "age", type = Integer.class) } ) ) @Entity public class Customer { // ... } ``` 然后,在询时使用 `@SqlResultSetMapping` 注解的名称来指定结果映射。例如: ```java String jpql = "SELECT c.name, c.age FROM Customer c"; TypedQuery<CustomObject> query = entityManager.createNativeQuery(jpql, "CustomObjectMapping"); List<CustomObject> resultList = query.getResultList(); ``` 3. 使用投影(Projection):你可以使用投影的方式来选择实体类的部分属性,并将它们映射到自定义对象中。例如: ```java String jpql = "SELECT c.name, c.age FROM Customer c"; TypedQuery<Object[]> query = entityManager.createQuery(jpql, Object[].class); List<Object[]> resultList = query.getResultList(); List<CustomObject> customObjects = resultList.stream() .map(objArr -> new CustomObject((String) objArr[0], (Integer) objArr[1])) .collect(Collectors.toList()); ``` 这些是几种常见的在 JPA返回自定义对象的方式。你可以根据具体的业务需求选择最适合你的方式。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值