网创优客建站品牌官网
为成都网站建设公司企业提供高品质网站建设
热线:028-86922220
成都专业网站建设公司

定制建站费用3500元

符合中小企业对网站设计、功能常规化式的企业展示型网站建设

成都品牌网站建设

品牌网站建设费用6000元

本套餐主要针对企业品牌型网站、中高端设计、前端互动体验...

成都商城网站建设

商城网站建设费用8000元

商城网站建设因基本功能的需求不同费用上面也有很大的差别...

成都微信网站建设

手机微信网站建站3000元

手机微信网站开发、微信官网、微信商城网站...

建站知识

当前位置:首页 > 建站知识

怎么在JPA中自定义对象接收查询结果集-创新互联

本篇文章给大家分享的是有关怎么在JPA中自定义对象接收查询结果集,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。

成都创新互联公司专注于企业营销型网站建设、网站重做改版、玉田网站定制设计、自适应品牌网站建设、H5建站商城网站建设、集团公司官网建设、成都外贸网站制作、高端网站制作、响应式网页设计等建站业务,价格优惠性价比高,为玉田等各大城市提供网站开发制作服务。
class Person { 
 @Id UUID id;
 String firstname, lastname;
 Address address; 
 static class Address {
 String zipCode, city, street;
 }
} 
interface PersonRepository extends Repository { 
 Collection findByLastname(String lastname);
}

自定义对象接收查询结果集方法如下:

(1)增加接收数据接口

interface NamesOnly { 
 String getFirstname();
 String getLastname();
}

(2)增加持久层接口

interface PersonRepository extends Repository { 
 Collection findByLastname(String lastname);
}

如果要对查询结果进行序列号的话就会有点问题:

{
 "errorCode": "00",
 "errorMessage": "操作成功",
 "returnObject": [
 {
  "createtime": 1526358195000,
  "id": 49,
  "lastupdatetime": 1526358195000,
  "status": "2",
  "target": {
  "createtime": 1526358195000,
  "lastupdatetime": 1526358195000,
  "check_Wicket": "1",
  "facility_name": "血压测量",
  "facility_Num": "C3",
  "id": 49,
  "status": "2",
  "check_name": "小汤154",
  "check_Num": "BY185201805140001"
  },
  "targetClass": "org.springframework.data.jpa.repository.query.AbstractJpaQuery$TupleConverter$TupleBackedMap"
 }
 ]
}

会出现targetClass这个字段,不能直接把结果拿来用,很恶心,又不想写代码中转下。

经过后来的摸索,其实如果只是为了返回JSON,也可以直接在Repository层直接用List>来返回,

Map对应单条查询结果,完美解决序列化问题。

完毕。就这么简单。

补充:SpringBoot JPA查询结果映射到自定义实体类

场景

举一个简单的例子:

比如有一个Position实体类

@Entity
@Table(name = "position")
public class Position implements Serializable {
 private static final long serialVersionUID = 768016840645708589L;
 @Id
 @GeneratedValue(strategy = GenerationType.IDENTITY)
 private Long id;
 private String name;
 private BigDecimal salary;
 private String city;
 
 ...//省略getset方法
 ...//省略toString方法 
}

然后有一个PositionDetail实体类

@Entity
@Table(name = "position_detail")
public class PositionDetail implements Serializable {
 private static final long serialVersionUID = 4556211410083251360L;
 @Id
 @GeneratedValue(strategy = GenerationType.IDENTITY)
 private Long id;
 private Long pid;
 private String description;
 
 ...//省略getset方法
 ...//省略toString方法 
}

需求:

查询职位基本信息,职位描述,因为涉及到两张表操作,简单的查询并不能满足我们的需求,因此就需要自定义查询接口并返回符合需求的结果。

接下来再定义一个实体类,用来接收查询结果

public class PositionDO {
 private Long id;
 private String name;
 private BigDecimal salary;
 private String city;
 private String description;
 public PositionDO(Long id, String name, BigDecimal salary, String city, String description) {
 this.id = id;
 this.name = name;
 this.salary = salary;
 this.city = city;
 this.description = description;
 }
 ...//省略getset方法
 ...//省略toString方法 
}

编写Dao接口,用来实现CRUD操作

public interface PositionDao extends JpaRepository {
 @Query(nativeQuery = true, value = "select t1.id, t1.name, t1.salary, t1.city, t2.description) \n" +
 "from position t1 left join position_detail t2 on t1.id = t2.pid \n" +
 "where t1.id = :id")
 PositionDO findPositionById(@Param("id") Long id);
}

思考:

如果这样写会不会出现问题?接下来我们编写一个测试类测试一下。

@SpringBootTest(classes = ShardingApplication.class)
@RunWith(SpringRunner.class)
public class TestShardingDatabase {
 @Resource
 PositionDao positionDao;
 @Test
 public void testQueryById() throws Exception{
 PositionDO positionDO = positionDao.findPositionById(548083053407240192L);
 System.out.println(positionDO);
 }
}

哈哈,翻车了吧,还好先测试了一波,问题不大。

Hibernate: select t1.id, t1.name, t1.salary, t1.city, t2.description 
from position t1 left join position_detail t2 on t1.id = t2.pid 
where t1.id = ?
org.springframework.core.convert.ConverterNotFoundException: No converter found capable of converting from type [org.springframework.data.jpa.repository.query.AbstractJpaQuery$TupleConverter$TupleBackedMap] to type [com.lsp.dto.PositionDO]

分析原因:

那么到底是为什么造成这样的原因呢?相信小伙伴们一眼就能看出来了,是因为Jpa找不到能够从类型转换的转换器,而抛出这样的异常。

现在问题来了,既然这样不行,那么我们想要实现映射到自定义结果集该如何实现呢?

实现:

JPA可以自定义SQL语句进行查询,然后查询语句可以通过原生SQL语句(原生SQL语句也就是在@Query注解里加上nativeQuery = true)进行查询。当然了,也可以通过JPQL进行查询。

我们这里就是通过JPQL进行查询,它的特征就是与原生SQL语句类似,完全面向对象,通过类名和属性访问,而不是表名和表属性。

由此PositionDao修改之后就像这样

public interface PositionDao extends JpaRepository {
 @Query(value = "select new com.lsp.domain.PositionDO(t1.id, t1.name, t1.salary, t1.city, t2.description) \n" +
  "from Position t1 left join PositionDetail t2 on t1.id = t2.pid \n" +
  "where t1.id = :id")
 PositionDO findPositionById(@Param("id") Long id);
}

接下来我们再运行测试方法看一下。

Hibernate: select position0_.id as col_0_0_, position0_.name as col_1_0_, position0_.salary as col_2_0_, position0_.city as col_3_0_, positionde1_.description as col_4_0_ from position position0_ left outer join position_detail positionde1_ on (position0_.id=positionde1_.pid) where position0_.id=?
PositionDO{id=548083053407240192, name='Jerry5', salary=10000.00, city='beijing5', description='this is message 5'}

以上就是怎么在JPA中自定义对象接收查询结果集,小编相信有部分知识点可能是我们日常工作会见到或用到的。希望你能通过这篇文章学到更多知识。更多详情敬请关注创新互联行业资讯频道。


分享名称:怎么在JPA中自定义对象接收查询结果集-创新互联
网址分享:http://bjjierui.cn/article/dosjsj.html

其他资讯