使用JPA自定義VO類(lèi)型轉(zhuǎn)換(EntityUtils工具類(lèi))
JPA自定義VO類(lèi)型轉(zhuǎn)換(EntityUtils工具類(lèi))
在JPA查詢(xún)中,如果需要返回自定義的類(lèi),可以使用EntityUtils工具類(lèi),該類(lèi)源碼:
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
/**
* @author 954L
* @create 2019/10/30 17:27
*/
public class EntityUtils {
private static final Logger log = LoggerFactory.getLogger(EntityUtils.class);
/**
* 將數(shù)組數(shù)據(jù)轉(zhuǎn)換為實(shí)體類(lèi)
* 此處數(shù)組元素的順序必須與實(shí)體類(lèi)構(gòu)造函數(shù)中的屬性順序一致
*
* @param list 數(shù)組對(duì)象集合
* @param clazz 實(shí)體類(lèi)
* @param <T> 實(shí)體類(lèi)
* @param model 實(shí)例化的實(shí)體類(lèi)
* @return 實(shí)體類(lèi)集合
*/
public static <T> List<T> castEntity(List<Object[]> list, Class<T> clazz, Object model) {
List<T> returnList = new ArrayList<T>();
if (list.isEmpty()) return returnList;
Object[] co = list.get(0);
List<Map> attributeInfoList = getFiledsInfo(model);
Class[] c2 = new Class[attributeInfoList.size()];
if (attributeInfoList.size() != co.length) {
return returnList;
}
for (int i = 0; i < attributeInfoList.size(); i++) {
c2[i] = (Class) attributeInfoList.get(i).get("type");
}
try {
for (Object[] o : list) {
Constructor<T> constructor = clazz.getConstructor(c2);
returnList.add(constructor.newInstance(o));
}
} catch (Exception ex) {
log.error("實(shí)體數(shù)據(jù)轉(zhuǎn)化為實(shí)體類(lèi)發(fā)生異常:異常信息:{}", ex.getMessage());
return returnList;
}
return returnList;
}
private static Object getFieldValueByName(String fieldName, Object modle) {
try {
String firstLetter = fieldName.substring(0, 1).toUpperCase();
String getter = "get" + firstLetter + fieldName.substring(1);
Method method = modle.getClass().getMethod(getter, new Class[]{});
Object value = method.invoke(modle, new Object[]{});
return value;
} catch (Exception e) {
return null;
}
}
private static List<Map> getFiledsInfo(Object model) {
Field[] fields = model.getClass().getDeclaredFields();
List<Map> list = new ArrayList(fields.length);
Map infoMap = null;
for (int i = 0; i < fields.length; i++) {
infoMap = new HashMap(3);
infoMap.put("type", fields[i].getType());
infoMap.put("name", fields[i].getName());
infoMap.put("value", getFieldValueByName(fields[i].getName(), model));
list.add(infoMap);
}
return list;
}
}
使用原生sql查詢(xún):
/**
* 根據(jù)公司名稱(chēng)查詢(xún)崗位
* @param name 公司名稱(chēng)
* @return List<EmploymentPosition>
*/
@Query(value = "select id, position, salary, people, experience, address, update_time from employment_position where company_name = ?1 and is_delete is false ORDER BY sort asc", nativeQuery = true)
List<Object[]> findByCompanyName(String name);
使用類(lèi)型轉(zhuǎn)換:
@Override
public List<EmploymentPositionVO> findByCompanyName(String name) {
List<Object[]> objects = employmentPositionRepository.findByCompanyName(name);
return EntityUtils.castEntity(objects, EmploymentPositionVO.class, new EmploymentPositionVO());
}
VO類(lèi)如下:
import lombok.Data;
import java.math.BigInteger;
import java.sql.Timestamp;
/**
* @website https://el-admin.vip
* @description /
* @author budezhenjia
* @date 2021-02-08
**/
@Data
public class EmploymentPositionVO {
/** ID */
private BigInteger id;
/** 崗位名稱(chēng) */
private String position;
/** 月薪 */
private String salary;
/** 人數(shù) */
private Integer people;
/** 工作經(jīng)驗(yàn) */
private String experience;
/** 工作地點(diǎn) */
private String address;
/** 更新時(shí)間 */
private Timestamp updateTime;
public EmploymentPositionVO(BigInteger id, String position, String salary, Integer people, String experience, String address, Timestamp updateTime) {
this.id = id;
this.position = position;
this.salary = salary;
this.people = people;
this.experience = experience;
this.address = address;
this.updateTime = updateTime;
}
public EmploymentPositionVO() {
}
}
注意!
查詢(xún)sql語(yǔ)句所查詢(xún)出來(lái)的字段應(yīng)與VO類(lèi)中屬性順序一致,類(lèi)型也需要一致!!
例如ID這個(gè)字段MySQL中類(lèi)型為bigint,VO類(lèi)中的類(lèi)型為bigInteger
dto,vo,po,bo等實(shí)體轉(zhuǎn)換工具類(lèi)
3層開(kāi)發(fā)以及不是多么新穎的開(kāi)發(fā)思想了,但是呢,苦于開(kāi)發(fā)的程序猿們,經(jīng)常會(huì)被各個(gè)實(shí)例之間的轉(zhuǎn)換弄得暈頭轉(zhuǎn)向,尤其是轉(zhuǎn)換的次數(shù)多了,一下就蒙了,不知道轉(zhuǎn)到哪里去了,博主也有這種困難,于是在網(wǎng)上到處找,找到了一些方法,并結(jié)合自身的開(kāi)發(fā)使用,填補(bǔ)一些坑,希望對(duì)大家有所幫助!
下面宣布這次的主角:dozer
他是誰(shuí),一看英文名就不懂吧,其實(shí)他是一個(gè)大家都知道的一個(gè)角色,spring里面他可是家喻戶曉的一個(gè)主角,沒(méi)錯(cuò)就是beanUtils(其實(shí),就是他的替身?。┲饕饔镁褪怯脕?lái)復(fù)制 JavaBean 屬性的類(lèi)庫(kù),什么叫復(fù)制,沒(méi)錯(cuò),就一模一樣的再來(lái)一份,但是這樣有一點(diǎn)點(diǎn)小小的區(qū)別,那就是,在使用的時(shí)候,需要指定一個(gè)“容器”(瞎說(shuō)的,就是一個(gè)映射接受對(duì)象,也可以叫做目標(biāo))來(lái)存放,不然,復(fù)制到哪去,是不是。
這個(gè)時(shí)候呀,就有一個(gè)經(jīng)紀(jì)人的出現(xiàn),需要通過(guò)“經(jīng)紀(jì)人”去代理復(fù)制,才能叫這個(gè)替身出來(lái)呀(專(zhuān)業(yè)替身30年,必須要有經(jīng)紀(jì)人)
<dependency>
<groupId>net.sf.dozer</groupId>
<artifactId>dozer</artifactId>
<version>5.5.1</version>
</dependency>
好了,經(jīng)紀(jì)人的名片已經(jīng)發(fā)出,這個(gè)時(shí)候,找到劇組,咱們看看這個(gè)替身能不能勝任,能應(yīng)用在哪些劇組,哪些場(chǎng)景!
第一種場(chǎng)景,完全一樣:(咦,度一樣了,那肯定好弄,又看不出來(lái)區(qū)別)
在各個(gè)實(shí)體是一樣的情況下,是很容易的,直接上不就行啦,也不用做啥處理是不是:
直接是用原始替身(API方法)
Mapper mapper = new DozerBeanMapper();
DestinationObject destObject =
mapper.map(sourceObject, DestinationObject.class);
嘿嘿,換一下衣服,直接OK了(也就是使用mapper轉(zhuǎn)換復(fù)制)
第二種場(chǎng)景,完全不一樣:
求乞,完全不一樣,這咋的弄呀,肯定得換替身,是不是!最起碼找個(gè)相似的嘛(不用擔(dān)心,咱們有化妝師呀,化妝走起?。?/p>
@Data
public class UserVo{
@Mapping("userName")
private String name;
@Mapping("password")
private String pwd;
}
看一下,是不是這個(gè)東西,好眼熟,沒(méi)錯(cuò),是咱們的vo
@Data
@TableName("user")
public class UserEntity implements Serializable {
@ApiModelProperty(value = "id")
@TableId(value = "id", type = IdType.INPUT)
private String id;
@ApiModelProperty(value = "用戶名")
@Mapping("name")
private String userName;
@ApiModelProperty(value = "密碼")
@Mapping("pwd")
private String password;
@ApiModelProperty(value = "登錄名")
private String loginName;
@ApiModelProperty(value = "創(chuàng)建時(shí)間")
private Date createTime;
@ApiModelProperty(value = "修改時(shí)間")
private Date updateTime;
@ApiModelProperty(value = "版本號(hào)")
private Integer version;
@ApiModelProperty(value = "作廢標(biāo)記")
private Integer deleted;
}
這個(gè)呢,是咱們的實(shí)體對(duì)象(也就是數(shù)據(jù)庫(kù)對(duì)象了)
看一下,是不是發(fā)小完全不一樣呀!
這里呢,@Mappin充當(dāng)了化妝品的角色,將每一個(gè)不同的細(xì)節(jié)進(jìn)行遮蓋,使其成為和原來(lái)的實(shí)例一模一樣(也就是做了映射了)
@Mapping("userName")
private String name;
上面就是一個(gè)化妝處理,將name映射為userName
每一個(gè)地方處理完成后,直接上劇組,看看能不能不被發(fā)現(xiàn)
/**
* 轉(zhuǎn)換實(shí)體為另一個(gè)指定的實(shí)體
* 任意一個(gè)參數(shù)為NULL時(shí) 會(huì)拋出NPE
*
* @param source 源實(shí)體 不能為NULL
* @param clazz 目標(biāo)實(shí)體 不能為NULL
* @param <T> 泛型
* @return 轉(zhuǎn)換后的結(jié)果
*/
@NonNull
public static <T> T convert(@NonNull Object source, @NonNull Class<T> clazz) {
return dozerMapper.map(source, clazz);
}
實(shí)例:
UserEntity userEntity = userMapper.selectOne(wrapper);
UserVo convert = null;
if(userEntity != null){
convert = Dozer.convert(userEntity, UserVo.class);
}
沒(méi)錯(cuò),這樣就完成轉(zhuǎn)換替身的操作了,結(jié)果是可以映射上的,各位可以試試哦!
第三種場(chǎng)景,一堆的不相同的替身(好難呀,一堆的不一樣,不能找?guī)讉€(gè)差不多的嗎?)
針對(duì)于這種一堆的不一樣的替身,dozer也有辦法,通過(guò)JAVA8的stream流來(lái)進(jìn)行化妝(化妝師牛,只能這么說(shuō))
/**
* 轉(zhuǎn)換List實(shí)體為另一個(gè)指定的實(shí)體
* source如果為NULL 會(huì)使用空集合
* 在目標(biāo)實(shí)體為NULL時(shí) 會(huì)拋出NPE
*
* @param source 源集合 可以為NULL
* @param clazz 目標(biāo)實(shí)體 不能為NULL
* @param <T> 泛型
* @return 轉(zhuǎn)換后的結(jié)果
*/
@Nullable
public static <T> List<T> convert(@Nullable List<?> source, @NonNull Class<T> clazz) {
return Optional.ofNullable(source)
.orElse(Collections.emptyList())
.stream()
.map(bean -> dozerMapper.map(bean, clazz))
.collect(Collectors.toList());
}
看見(jiàn)沒(méi),這樣一弄,就OK了,咦,是不是發(fā)現(xiàn)化妝師dozerMapper從哪來(lái)的(這么牛的化妝師,召幾個(gè)開(kāi)化妝店去),博主告訴你,這個(gè)化妝師呀,并不是越多也好的(大家都知道,一樣的對(duì)象,建立太多浪費(fèi)空間),只需要建立一個(gè)全局唯一的就行了,博主帶你們看一下化妝間就明白了(工具類(lèi)來(lái)了)
package cn.yichehuoban.ycbb.platform.util.beanutils;
import org.dozer.Mapper;
import org.springframework.lang.NonNull;
import org.springframework.lang.Nullable;
import org.springframework.stereotype.Component;
import java.util.Collections;
import java.util.List;
import java.util.Optional;
import java.util.stream.Collectors;
@Component
public class Dozer {
/**
* dozer轉(zhuǎn)換的核心mapper對(duì)象
*/
public static final Mapper dozerMapper = new org.dozer.DozerBeanMapper();
/**
* 轉(zhuǎn)換實(shí)體為另一個(gè)指定的實(shí)體
* 任意一個(gè)參數(shù)為NULL時(shí) 會(huì)拋出NPE
*
* @param source 源實(shí)體 不能為NULL
* @param clazz 目標(biāo)實(shí)體 不能為NULL
* @param <T> 泛型
* @return 轉(zhuǎn)換后的結(jié)果
*/
@NonNull
public static <T> T convert(@NonNull Object source, @NonNull Class<T> clazz) {
return dozerMapper.map(source, clazz);
}
/**
* 轉(zhuǎn)換List實(shí)體為另一個(gè)指定的實(shí)體
* source如果為NULL 會(huì)使用空集合
* 在目標(biāo)實(shí)體為NULL時(shí) 會(huì)拋出NPE
*
* @param source 源集合 可以為NULL
* @param clazz 目標(biāo)實(shí)體 不能為NULL
* @param <T> 泛型
* @return 轉(zhuǎn)換后的結(jié)果
*/
@Nullable
public static <T> List<T> convert(@Nullable List<?> source, @NonNull Class<T> clazz) {
return Optional.ofNullable(source)
.orElse(Collections.emptyList())
.stream()
.map(bean -> dozerMapper.map(bean, clazz))
.collect(Collectors.toList());
}
}
在給大家看一下我們的替身
@Data
public class UserVo{
@Mapping("userName")
private String name;
@Mapping("password")
private String pwd;
private String loginName;
private Integer version;
private Integer deleted;
private String id;
}
一樣的地方直接就映射上了,不一樣的地方使用 @Mapping注解,填寫(xiě)上源指定對(duì)象的字段名就行
這只是幾種方法,還有其他雙向映射,數(shù)據(jù)拷貝等,可以看看他的官方文檔dozer
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java Swing JToggleButton開(kāi)關(guān)按鈕的實(shí)現(xiàn)
這篇文章主要介紹了Java Swing JToggleButton開(kāi)關(guān)按鈕的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-12-12
Java 根據(jù)貸款年限對(duì)應(yīng)利率計(jì)算功能實(shí)現(xiàn)解析
這篇文章主要介紹了Java 根據(jù)貸款年限對(duì)應(yīng)利率計(jì)算功能實(shí)現(xiàn)解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-10-10
SpringBoot2.x 整合 thumbnailator 圖片處理的示例代碼
這篇文章主要介紹了SpringBoot2.x 之整合 thumbnailator 圖片處理,本文通過(guò)示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-10-10
springboot多模塊多環(huán)境配置文件問(wèn)題(動(dòng)態(tài)配置生產(chǎn)和開(kāi)發(fā)環(huán)境)
這篇文章主要介紹了springboot多模塊多環(huán)境配置文件問(wèn)題(動(dòng)態(tài)配置生產(chǎn)和開(kāi)發(fā)環(huán)境),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04
SpringBoot大學(xué)心理服務(wù)系統(tǒng)實(shí)現(xiàn)流程分步講解
本系統(tǒng)主要論述了如何使用JAVA語(yǔ)言開(kāi)發(fā)一個(gè)大學(xué)生心理服務(wù)系統(tǒng) ,本系統(tǒng)將嚴(yán)格按照軟件開(kāi)發(fā)流程進(jìn)行各個(gè)階段的工作,采用B/S架構(gòu),面向?qū)ο缶幊趟枷脒M(jìn)行項(xiàng)目開(kāi)發(fā)2022-09-09
Java消息隊(duì)列的簡(jiǎn)單實(shí)現(xiàn)代碼
本篇文章主要介紹了Java消息隊(duì)列的簡(jiǎn)單實(shí)現(xiàn)代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-07-07

