Java 类动态添加属性字段的操作

 更新时间:2021年02月18日 14:45:09   作者:前尘忆梦  
这篇文章主要介绍了Java 类动态添加属性字段的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧

说明:

做项目中遇到一种场景,需要根据查询时间段, 获取时间段中中每个月份对应的金额(费用统计)。

如截图中的两列

因为列是动态的, 首先想到的就是后天拼接JSON格式字符串, 然后返回到前台, 组装表头及内容。

但是当前系统中easyUI版本为1.2,并不支持 data属性(官方从1.3.2开始支持)。所以只能返回list<T> 格式。

网上一搜相关代码很多, 看客可以自己搜索一下。 我这里记录一下我当时使用场景及用法,已备以后使用。

1.需要引用cglib jar包, 我用的版本是2.2

2.建一个实体对象 DynamicBean.java 。主要用来处理对象。

public class DynamicBean {
 private Object object = null; // 动态生成的类 
 private BeanMap beanMap = null; // 存放属性名称以及属性的类型 
 public DynamicBean() {
  super();
 }
 
 public DynamicBean(Map propertyMap) {
  this.object = generateBean(propertyMap);
  this.beanMap = BeanMap.create(this.object);
 }
 
 /**
  * @param propertyMap
  * @return
  */
 private Object generateBean(Map propertyMap) {
  BeanGenerator generator = new BeanGenerator();
  Set keySet = propertyMap.keySet();
  for (Iterator<String> i = keySet.iterator(); i.hasNext();) {
   String key = (String) i.next();
   generator.addProperty(key, (Class) propertyMap.get(key));
  }
  return generator.create();
 }
 
 /**
  * ��bean���Ը�ֵ
  * @param property ������
  * @param value ֵ
  */
 public void setValue(Object property, Object value) {
  beanMap.put(property, value);
 }
 
 /**
  * ͨ���������õ�����ֵ
  * @param property ������
  * @return ֵ
  */
 public Object getValue(String property) {
  return beanMap.get(property);
 }
 
 /**
  * 返回新生成的对象
  * @return
  */
 public Object getObject() {
  return this.object;
 }
}

3. 原来对象, 及需要拼接到对象中的属性字段集合处理方法。

/**
*参数说明:
* object : 查询结果数组中对象。
* moneyMap : 为对象对应所有月份数据集合
* 解释:已经查询出一组账单对象集合List<Bill> , 而moneyMap为对象中的一个属性 
* Map<String,Bigdecimal>, 存放了月份及金额
*/
private Object dynamicClass(Object object, Map<String, BigDecimal> moneyMap) throws Exception {
  // 字段 - 值 集合
  HashMap<String, Object> returnMap = new HashMap<String, Object>();
  // 字段 - 字段类型 集合
  HashMap<String, Object> typeMap = new HashMap<String, Object>();
  // 获取传入类
  Class<? extends Object> type = object.getClass();
  BeanInfo beanInfo = Introspector.getBeanInfo(type);
  PropertyDescriptor[] propertyDescriptors = beanInfo.getPropertyDescriptors();
  // 获取对象中已存在的数据
  for (int i = 0; i < propertyDescriptors.length; i++) {
   PropertyDescriptor descriptor = propertyDescriptors[i];
   String propertyName = descriptor.getName();
   if (!propertyName.equals("class") && !propertyName.equals("monthMap")) {
    Method readMethod = descriptor.getReadMethod();
    Object result = readMethod.invoke(object, new Object[0]);
    if (result != null) {
     returnMap.put(propertyName, result);
    } else {
     String propertyType = descriptor.getPropertyType().toString();
     if (propertyType.contains("java.math.BigDecimal")) {
      returnMap.put(propertyName, new BigDecimal(0));
     } else {
      returnMap.put(propertyName, "");
     }
    }
    typeMap.put(propertyName, descriptor.getPropertyType());
   }
  }
  // 获取月份数据, 变为字段属性
  Set<String> monthKeys = moneyMap.keySet();
  for (Iterator<String> it = monthKeys.iterator(); it.hasNext();) {
   String key = (String) it.next();
   // 字段类型
   typeMap.put(key, Class.forName("java.math.BigDecimal"));
   // 字段对应值
   returnMap.put(key, moneyMap.get(key));
  }
  // map转换成实体对象
  DynamicBean bean = new DynamicBean(typeMap);
  // 赋值
  Set<String> keys = typeMap.keySet();
  for (Iterator<String> it = keys.iterator(); it.hasNext();) {
   String key = (String) it.next();
   bean.setValue(key, returnMap.get(key));
  }
  Object obj = bean.getObject();
  return obj;
 }

做笔记使用, 说不定以后还会用到。

补充:java动态的生成类的属性、并赋值

1、springboot项目中,在build.gradle中,配置jar包

compile("commons-beanutils:commons-beanutils:1.9.3")
compile("cglib:cglib-nodep:3.2.4")

2、创建DynamicBean

import java.util.Map; 
import net.sf.cglib.beans.BeanGenerator;
import net.sf.cglib.beans.BeanMap; 
public class DynamicBean {
 
 /**
  * 目标对象
  */
 private Object target;
 /**
  * 属性集合
  */
 private BeanMap beanMap;
 
 public DynamicBean(Class superclass, Map<String, Class> propertyMap){
  this.target = generateBean(superclass, propertyMap);
  this.beanMap = BeanMap.create(this.target);
 }
 
 /**
  * bean 添加属性和值
  *
  * @param property
  * @param value
  */
 public void setValue(String property, Object value) {
  beanMap.put(property, value);
 }
 
 /**
  * 获取属性值
  *
  * @param property
  * @return
  */
 public Object getValue(String property) {
  return beanMap.get(property);
 }
 
 /**
  * 获取对象
  *
  * @return
  */
 public Object getTarget() {
  return this.target;
 }
 
 /**
  * 根据属性生成对象
  *
  * @param superclass
  * @param propertyMap
  * @return
  */
 private Object generateBean(Class superclass, Map<String, Class> propertyMap) {
  BeanGenerator generator = new BeanGenerator();
  if (null != superclass) {
   generator.setSuperclass(superclass);
  }
  BeanGenerator.addProperties(generator, propertyMap);
  return generator.create();
 }
}

3、创建ReflecUtil转换的工具类

import java.beans.PropertyDescriptor;
import java.util.HashMap;
import java.util.Map; 
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger; 
import org.apache.commons.beanutils.PropertyUtilsBean; 
import com.sunxung.factoring.entity.DynamicBean;
import com.sunxung.factoring.entity.attendanceManagement.AttendanceVo;
 
/**
 * @className:ReflectUtil
 * @description:动态生成类的属性、并且赋值
 * @date:2018年4月3日 下午2:33:10
 */
public class ReflectUtil {
 
 static Logger logger = LogManager.getLogger(ReflectUtil.class);
 
 @SuppressWarnings("rawtypes")
 public static Object getTarget(Object dest, Map<String, Object> addProperties) {
  PropertyUtilsBean propertyUtilsBean = new PropertyUtilsBean();
  PropertyDescriptor[] descriptors = propertyUtilsBean.getPropertyDescriptors(dest);
  Map<String, Class> propertyMap = new HashMap<>();
  for (PropertyDescriptor d : descriptors) {
   if (!"class".equalsIgnoreCase(d.getName())) {
    propertyMap.put(d.getName(), d.getPropertyType());
   }
  }
  // add extra properties
  addProperties.forEach((k, v) -> propertyMap.put(k, v.getClass()));
  // new dynamic bean
  DynamicBean dynamicBean = new DynamicBean(dest.getClass(), propertyMap);
  // add old value
  propertyMap.forEach((k, v) -> {
   try {
    // filter extra properties
    if (!addProperties.containsKey(k)) {
     dynamicBean.setValue(k, propertyUtilsBean.getNestedProperty(dest, k));
    }
   } catch (Exception e) {
    logger.error(e.getMessage(), e);
   }
  });
  // add extra value
  addProperties.forEach((k, v) -> {
   try {
    dynamicBean.setValue(k, v);
   } catch (Exception e) {
    logger.error(e.getMessage(), e);
   }
  });
  Object target = dynamicBean.getTarget();
  return target;
 }
 
 public static void main(String[] args) {
  AttendanceVo entity = new AttendanceVo();
  Map<String, Object> addProperties = new HashMap<>();
  addProperties.put("day31", "你好");
  AttendanceVo newVo = (AttendanceVo) getTarget(entity, addProperties);
  System.out.println(newVo.getDay0());
 }
}

4、在项目中动态生成属性并且赋值的使用

private AttendanceVo autoDetailNew(SearchAttendanceVo search, AttendanceVo att) {
  search.setDingdingUserId(att.getDingdingUserId());
  List<Attendance> detailList = attendanceMapper.findDetailList(search);
  Map<String, Object> addProperties = new HashMap<>();
  for (Attendance attendance : detailList) {
   addProperties.put("day" + DateUtil.getDayStringFormatYMD(attendance.getAttendanceDate()),
     attendance.getRemark());
  }
  AttendanceVo newVo = (AttendanceVo) ReflectUtil.getTarget(att, addProperties);
  return newVo;
 }

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。如有错误或未考虑完全的地方,望不吝赐教。

相关文章

  • 关于集合和字符串的互转实现方法

    关于集合和字符串的互转实现方法

    下面小编就为大家带来一篇关于集合和字符串的互转实现方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-08-08
  • Java匿名内部类导致内存泄露的原因与解决方案详解

    Java匿名内部类导致内存泄露的原因与解决方案详解

    这篇文章主要为大家详细介绍了Java因为匿名内部类导致内存泄露的原因以及其解决方案,文中的示例代码讲解详细,希望对大家有所帮助
    2022-11-11
  • 快速上手Mybatis-plus结构构建过程

    快速上手Mybatis-plus结构构建过程

    这篇文章主要介绍了快速上手Mybatis-plus结构构建过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-07-07
  • Java8 Optional优雅空值判断的示例代码

    Java8 Optional优雅空值判断的示例代码

    这篇文章主要介绍了Java8 Optional优雅空值判断的相关知识,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-05-05
  • java堆栈类使用实例(java中stack的使用方法)

    java堆栈类使用实例(java中stack的使用方法)

    java中stack的使用方法,堆栈是一种"后进先出"(LIFO) 的数据结构, 只能在一端进行插入(称为"压栈") 或删除 (称为"出栈")数据的操作,下面看示例吧
    2013-12-12
  • 使用maven创建普通项目命令行程序详解

    使用maven创建普通项目命令行程序详解

    大部分使用maven创建的是web项目,这里使用maven创建一个命令行程序,目的是让大家了解maven特点和使用方式,有需要的朋友可以借鉴参考下
    2021-10-10
  • 使用Swagger时Controller中api接口显示不全的问题分析及解决

    使用Swagger时Controller中api接口显示不全的问题分析及解决

    swagger是一个十分好用的api接口管理、测试框架,现在越来越多的人使用这个做接口的测试和管理,但经常遇到Controller中的api接口显示不全的问题,所以本文给大家详细分析了问题以及解决方法,需要的朋友可以参考下
    2024-02-02
  • SpringMVC如何域对象共享数据

    SpringMVC如何域对象共享数据

    在Spring MVC中,可以使用域对象来共享数据,域对象是一个Map类型的对象,可以在请求处理方法之间共享数据,本文给大家介绍SpringMVC 域对象共享数据的示例代码,一起看看吧
    2023-09-09
  • SpringBoot整合EasyExcel实现Excel表格导出功能

    SpringBoot整合EasyExcel实现Excel表格导出功能

    这篇文章主要介绍了SpringBoot整合EasyExcel实现Excel表格导出功能,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的朋友可以参考一下
    2022-07-07
  • Java设计模式之原型模式详解

    Java设计模式之原型模式详解

    这篇文章主要介绍了Java设计模式之原型模式详解,文中有非常详细的代码示例,对正在学习java的小伙伴们有非常好的帮助,需要的朋友可以参考下
    2021-05-05

最新评论