JPA自定义对象接收查询结果集操作
最近使用JPA的时候,碰到需要自定义查询结果集的场景,网上搜了一下,都是需要自定义方法写一大串代码实现的,太繁琐了,有那时间还不如用mybaits。
用JPA就是要尽量通过声明接口解决持久层问题,要不然鬼用。逼得没办法去了官网看看文档,再没有就放弃了,没时间看源码。最终找到我想要的结果了。
例如,传统的JPA接口实现如下所示:
classPerson{ @IdUUIDid; Stringfirstname,lastname; Addressaddress; staticclassAddress{ StringzipCode,city,street; } } interfacePersonRepositoryextendsRepository{ Collection findByLastname(Stringlastname); }
自定义对象接收查询结果集方法如下:
(1)增加接收数据接口
interfaceNamesOnly{ StringgetFirstname(); StringgetLastname(); }
(2)增加持久层接口
interfacePersonRepositoryextendsRepository{ Collection findByLastname(Stringlastname); }
如果要对查询结果进行序列号的话就会有点问题:
{ "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
完毕。就这么简单。
补充:SpringBootJPA查询结果映射到自定义实体类
场景
举一个简单的例子:
比如有一个Position实体类
@Entity @Table(name="position") publicclassPositionimplementsSerializable{ privatestaticfinallongserialVersionUID=768016840645708589L; @Id @GeneratedValue(strategy=GenerationType.IDENTITY) privateLongid; privateStringname; privateBigDecimalsalary; privateStringcity; ...//省略getset方法 ...//省略toString方法 }
然后有一个PositionDetail实体类
@Entity @Table(name="position_detail") publicclassPositionDetailimplementsSerializable{ privatestaticfinallongserialVersionUID=4556211410083251360L; @Id @GeneratedValue(strategy=GenerationType.IDENTITY) privateLongid; privateLongpid; privateStringdescription; ...//省略getset方法 ...//省略toString方法 }
需求:
查询职位基本信息,职位描述,因为涉及到两张表操作,简单的查询并不能满足我们的需求,因此就需要自定义查询接口并返回符合需求的结果。
接下来再定义一个实体类,用来接收查询结果
publicclassPositionDO{ privateLongid; privateStringname; privateBigDecimalsalary; privateStringcity; privateStringdescription; publicPositionDO(Longid,Stringname,BigDecimalsalary,Stringcity,Stringdescription){ this.id=id; this.name=name; this.salary=salary; this.city=city; this.description=description; } ...//省略getset方法 ...//省略toString方法 }
编写Dao接口,用来实现CRUD操作
publicinterfacePositionDaoextendsJpaRepository{ @Query(nativeQuery=true,value="selectt1.id,t1.name,t1.salary,t1.city,t2.description)\n"+ "frompositiont1leftjoinposition_detailt2ont1.id=t2.pid\n"+ "wheret1.id=:id") PositionDOfindPositionById(@Param("id")Longid); }
思考:
如果这样写会不会出现问题?接下来我们编写一个测试类测试一下。
@SpringBootTest(classes=ShardingApplication.class) @RunWith(SpringRunner.class) publicclassTestShardingDatabase{ @Resource PositionDaopositionDao; @Test publicvoidtestQueryById()throwsException{ PositionDOpositionDO=positionDao.findPositionById(548083053407240192L); System.out.println(positionDO); } }
哈哈,翻车了吧,还好先测试了一波,问题不大。
Hibernate:selectt1.id,t1.name,t1.salary,t1.city,t2.description frompositiont1leftjoinposition_detailt2ont1.id=t2.pid wheret1.id=? org.springframework.core.convert.ConverterNotFoundException:Noconverterfoundcapableofconvertingfromtype[org.springframework.data.jpa.repository.query.AbstractJpaQuery$TupleConverter$TupleBackedMap]totype[com.lsp.dto.PositionDO]
分析原因:
那么到底是为什么造成这样的原因呢?相信小伙伴们一眼就能看出来了,是因为Jpa找不到能够从类型转换的转换器,而抛出这样的异常。
现在问题来了,既然这样不行,那么我们想要实现映射到自定义结果集该如何实现呢?
实现:
JPA可以自定义SQL语句进行查询,然后查询语句可以通过原生SQL语句(原生SQL语句也就是在@Query注解里加上nativeQuery=true)进行查询。当然了,也可以通过JPQL进行查询。
我们这里就是通过JPQL进行查询,它的特征就是与原生SQL语句类似,完全面向对象,通过类名和属性访问,而不是表名和表属性。
由此PositionDao修改之后就像这样
publicinterfacePositionDaoextendsJpaRepository{ @Query(value="selectnewcom.lsp.domain.PositionDO(t1.id,t1.name,t1.salary,t1.city,t2.description)\n"+ "fromPositiont1leftjoinPositionDetailt2ont1.id=t2.pid\n"+ "wheret1.id=:id") PositionDOfindPositionById(@Param("id")Longid); }
接下来我们再运行测试方法看一下。
Hibernate:selectposition0_.idascol_0_0_,position0_.nameascol_1_0_,position0_.salaryascol_2_0_,position0_.cityascol_3_0_,positionde1_.descriptionascol_4_0_frompositionposition0_leftouterjoinposition_detailpositionde1_on(position0_.id=positionde1_.pid)whereposition0_.id=? PositionDO{id=548083053407240192,name='Jerry5',salary=10000.00,city='beijing5',description='thisismessage5'}
ok了,结果已经正确查询出来。
总结:
注意上面的SQL语句是面向对象的,对应的字段也都是实体类里面的属性。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持毛票票。如有错误或未考虑完全的地方,望不吝赐教。