Java动态为类添加属性并赋值

 1.创建一个类,用来接收动态添加属性后的一些值

package com.yunzhi.village.test;

import lombok.Data;
import org.springframework.cglib.beans.BeanGenerator;
import org.springframework.cglib.beans.BeanMap;

import java.util.Iterator;
import java.util.Map;
import java.util.Set;

@Data
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();
    }

    /**
     * set beanMap
     * @param property
     * @param value ֵ
     */
    public void setValue(Object property, Object value) {
        beanMap.put(property, value);
    }

    /**
     * ͨget beanMap
     * @param property
     * @return ֵ
     */
    public Object getValue(String property) {
        return beanMap.get(property);
    }
}

2.业务逻辑代码

moneyMap:代表业务中需要动态添加的属性(属性名和值)

 /**
     *参数说明:
     * object : 查询结果数组中对象。
     * moneyMap : 为对象对应所有月份数据集合
     * Map<String,Bigdecimal>, 存放了月份及金额
     */
    private Object dynamicClass(Object object, Map<String, BigDecimal> moneyMap) throws Exception {
        // 字段 - 值 集合
        HashMap<String, Object> returnMap = new HashMap<>();
        // 字段 - 字段类型 集合
        HashMap<String, Object> typeMap = new HashMap<>();
        // 获取传入类
        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;
    }

3.执行即可

   public static void main(String[] args) throws Exception {
        Test test = new Test();
        User user = new User("ddd","温州","","男",12);
        Map<String, BigDecimal> map = new HashMap<>();
        map.putIfAbsent("2",  new BigDecimal(45678));
        map.putIfAbsent("4",  new BigDecimal(12365));
        Object o = test.dynamicClass(user, map);
    }

此处的 o 对象,是把user中的属性和动态添加的 BigDecimal 2 = 45678 和  BigDecimal 4 = 45678整合到o对象中。

4.涉及的Spring技术(后续可能会出Spring源码讲解的文章)

BeanInfo beanInfo = Introspector.getBeanInfo(type);
PropertyDescriptor[] propertyDescriptors = beanInfo.getPropertyDescriptors();
BeanMap beanMap;
BeanGenerator generator = new BeanGenerator();
  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
在使用cglib动态给对象添加属性之前,需要先了解一下Java的反射机制。 Java反射机制是指在运行状态中,对于任意一个,都能够知道这个的所有属性和方法;并且能够调用这个的任意方法,获取任意属性。 cglib是一个强大的代码生成库,它是基于ASM框架实现的,可以在运行时动态生成新的。cglib通过继承目标对象实现代理,并在子添加新的属性和方法。 下面是一个例子,演示了如何使用cglib动态给对象添加属性赋值: ```java import net.sf.cglib.beans.BeanGenerator; public class DynamicAddPropertyDemo { public static void main(String[] args) { // 创建一个BeanGenerator对象 BeanGenerator generator = new BeanGenerator(); // 设置要创建的对象的属性 generator.addProperty("name", String.class); generator.addProperty("age", Integer.class); // 创建一个对象 Object obj = generator.create(); // 给对象赋值 ReflectUtil.setProperty(obj, "name", "张三"); ReflectUtil.setProperty(obj, "age", 20); // 输出对象的属性 System.out.println("name:" + ReflectUtil.getProperty(obj, "name")); System.out.println("age:" + ReflectUtil.getProperty(obj, "age")); } } class ReflectUtil { /** * 获取对象的属性值 */ public static Object getProperty(Object obj, String propertyName) { try { BeanMap map = BeanMap.create(obj); return map.get(propertyName); } catch (Exception e) { e.printStackTrace(); return null; } } /** * 设置对象的属性值 */ public static void setProperty(Object obj, String propertyName, Object value) { try { BeanMap map = BeanMap.create(obj); map.put(propertyName, value); } catch (Exception e) { e.printStackTrace(); } } } ``` 运行结果: ``` name:张三 age:20 ``` 上面的例子中,首先创建了一个BeanGenerator对象,然后使用addProperty()方法添加了两个属性:name和age。 接着使用create()方法创建了一个对象,并调用ReflectUtil中的setProperty()方法给对象的属性赋值。最后,使用ReflectUtil中的getProperty()方法获取对象的属性值并输出。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值