1.问题介绍
在我们使用springdatajpa时如果查询所有字段我们就可以用这个表对应的实体类来直接接收数据返回,
但是当我们查询多表关联查询的字段无法用实体类直接接收时,我们就需要自己封装
2.例如
@Query(value = "SELECT a.iar01,a.aud01,d.aud02,d.aud04,a.ark02,a.ark03 FROM aud_ark a JOIN rcp_aud d ON a.aud01 = d.aud01 "
+ " where iar01=?1 and aud04= ?2 order by a.ark03 desc",nativeQuery = true)
List<Object []> GetArkDetailList(String iar01,Integer aud04);
这里我们就关联了多表,查询不同的字段,然后我们接收 返回的是List<Object []>类型这种数据我们不好处理,所以接下来我们就来解决这个问题
3.添加EntityUtils 工具类
我们需要这个工具类进行转化
package cn.eracare.utils;
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;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class EntityUtils {
private static Logger logger = LoggerFactory.getLogger(EntityUtils.class);
/**
* 将数组数据转换为实体类
* 此处数组元素的顺序必须与实体类构造函数中的属性顺序一致
*
* @param list 数组对象集合
* @param clazz 实体类
* @param <T> 实体类
* @param model 实例化的实体类
* @return 实体类集合
*/
@SuppressWarnings("rawtypes")
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) {
logger.error("实体数据转化为实体类发生异常:异常信息:{}", ex.getMessage());
return returnList;
}
return returnList;
}
/**
* 根据属性名获取属性值
*
* @param fieldName 属性名
* @param modle 实体类
* @return 属性值
*/
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;
}
}
/**
* 获取属性类型(type),属性名(name),属性值(value)的map组成的list
*
* @param model 实体类
* @return list集合
*/
@SuppressWarnings(value={"unchecked", "rawtypes"})
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;
}
}
4.创建返回实体类
在使用工具类之前我们需要建立好返回实体类例如:
/**
*
*/
package cn.eracare.vo;
import java.math.BigDecimal;
import lombok.Data;
/**
* <p>Title: GetArkDetailListVo.java</p>
* <p>Description:添加注释 </p>
* @author youthMing
* @date 2020年3月31日
*/
@Data
public class GetArkDetailListVo {
private String iar01;
private String aud01;
private String aud02;
private Integer aud04;
private BigDecimal ark02;
private BigDecimal ark03;
/**
* <p>Title: GetArkDetailListVo.java</p>
* <p>Description:添加注释 </p>
* @author youthMing
* @date 2020年3月31日
*/
public GetArkDetailListVo(String iar01, String aud01, String aud02, Integer aud04, BigDecimal ark02,
BigDecimal ark03) {
super();
this.iar01 = iar01;
this.aud01 = aud01;
this.aud02 = aud02;
this.aud04 = aud04;
this.ark02 = ark02;
this.ark03 = ark03;
}
/**
* <p>Title: GetArkDetailListVo.java</p>
* <p>Description:添加注释 </p>
* @author youthMing
* @date 2020年3月31日
*/
public GetArkDetailListVo() {
super();
}
}
注意除了要实现get,set方法还需要加上两个构造方法
5.使用工具类
//获得查询所得集合
List<Object[]> getArkDetailList = audArkRepository.GetArkDetailList(iar01, aud04);
//使用工具类(这里的参数GetArkDetailListVo是返回的实体类,在上面有详解)
List<GetArkDetailListVo> castEntity =
EntityUtils.castEntity(getArkDetailList, GetArkDetailListVo.class, new GetArkDetailListVo());
最后使用工具类之后我们就可以得到需要的返回实体类,返回给前台。
注意事项
返回实体类要和查询字段的位置,字段类型一模一样,否则将无法封装。