JPA自定義對象接收查詢結果集操作
最近使用JPA的時候,碰到需要自定義查詢結果集的場景,網上搜瞭一下,都是需要自定義方法寫一大串代碼實現的,太繁瑣瞭,有那時間還不如用mybaits。
用JPA就是要盡量通過聲明接口解決持久層問題,要不然鬼用。逼得沒辦法去瞭官網看看文檔,再沒有就放棄瞭,沒時間看源碼。最終找到我想要的結果瞭。
例如,傳統的JPA接口實現如下所示:
class Person { @Id UUID id; String firstname, lastname; Address address; static class Address { String zipCode, city, street; } } interface PersonRepository extends Repository<Person, UUID> { Collection<Person> findByLastname(String lastname); }
自定義對象接收查詢結果集方法如下:
(1)增加接收數據接口
interface NamesOnly { String getFirstname(); String getLastname(); }
(2)增加持久層接口
interface PersonRepository extends Repository<Person, UUID> { Collection<NamesOnly> 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<String,Object>>來返回,
Map<String,Object>對應單條查詢結果,完美解決序列化問題。
完畢。就這麼簡單。
補充: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<Position, Long> { @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<Position, Long> { @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'}
ok瞭,結果已經正確查詢出來。
總結:
註意上面的SQL語句是面向對象的,對應的字段也都是實體類裡面的屬性。
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。如有錯誤或未考慮完全的地方,望不吝賜教。
推薦閱讀:
- Springboot JPA如何使用distinct返回對象
- 使用JPA自定義SQL查詢結果
- Spring Data JPA 在 @Query 中使用投影的方法示例詳解
- spring data jpa 創建方法名進行簡單查詢方式
- SpringData如何通過@Query註解支持JPA語句和原生SQL語句