Java 动态为类添加属性

“衰兰送客咸阳道,天若有情天亦老”

有一个这样的功能:展示一个报表,表头是动态的,可以手动新增或删除,也就是说会修改表结构,新增或删除表字段。然而一般实体类都是对应表结构的,那么如何给实体增加属性呢?虽然我自己的解决方案是直接返回List<map>,然后让对应的列适配到map的key中。但是这里说说给实体类动态添加属性。

实现代码


import com.google.common.collect.Maps;
import net.sf.cglib.beans.BeanGenerator;
import net.sf.cglib.beans.BeanMap;
import org.apache.commons.beanutils.PropertyUtilsBean;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Map;

/**
 *@author 
 *@date 2019/10/9
 *@description
 */
public class ReflectUtil {

    static Logger logger = LoggerFactory.getLogger(ReflectUtil.class);

    public static Object getTarget(Object dest, Map<String, Object> addProperties) {
        // get property map
        PropertyUtilsBean propertyUtilsBean = new PropertyUtilsBean();
        PropertyDescriptor[] descriptors = propertyUtilsBean.getPropertyDescriptors(dest);
        Map<String, Class> propertyMap = Maps.newHashMap();
        for (PropertyDescriptor d : descriptors) {
            if (!"class".equalsIgnoreCase(d.getName())) {
                propertyMap.put(d.getName(), d.getPropertyType());
            }
        }
        // add extra properties
        for (Map.Entry<String, Object> entry : addProperties.entrySet()) {
            propertyMap.put(entry.getKey(), entry.getValue().getClass());
        }
//        addProperties.forEach((k, v) -> propertyMap.put(k, v.getClass()));
        // new dynamic bean
        DynamicBean dynamicBean = new DynamicBean(dest.getClass(), propertyMap);
        // add old value
        for (Map.Entry<String, Class> entry : propertyMap.entrySet()) {
            try {
                // filter extra properties
                if (!addProperties.containsKey(entry.getKey())) {
                    dynamicBean.setValue(entry.getKey(), propertyUtilsBean.getNestedProperty(dest, entry.getKey()));
                }
            } catch (Exception e) {
                logger.error(e.getMessage(), e);
            }
        }
        ;
        // add extra value
        for (Map.Entry<String, Object> entry : addProperties.entrySet()) {
            try {
                dynamicBean.setValue(entry.getKey(), entry.getValue());
            } catch (Exception e) {
                logger.error(e.getMessage(), e);
            }
        }
        ;
        Object target = dynamicBean.getTarget();
        return target;
    }

    public static 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();
        }
    }

    public static void main(String[] args) {
        TestClass entity = new TestClass();
        entity.setAge(15);
        entity.setId(1);
        entity.setName("张三");
        //动态设置属性字段
        Map<String, Object> addProperties = new HashMap() {{
            put("sex", "男");
            put("telephone", "123456789");
        }};
        //获取实体中给属性的值
        Object target = getTarget(entity, addProperties);
        System.out.println(getFieldValueByName("id", target));
        System.out.println(getFieldValueByName("name", target));
        System.out.println(getFieldValueByName("age", target));
        System.out.println(getFieldValueByName("sex", target));
        System.out.println(getFieldValueByName("telephone", target));
    }

    //获取实体中属性的值
    private static Object getFieldValueByName(String fieldName, Object o) {
        try {
            //通过反射获取值
            String firstLetter = fieldName.substring(0, 1).toUpperCase();
            String getter = "get" + firstLetter + fieldName.substring(1);
            Method method = o.getClass().getMethod(getter, new Class[]{});
            Object value = method.invoke(o, new Object[]{});
            return value;
        } catch (Exception e) {
            logger.error(e.getMessage(), e);
            return null;
        }
    }

    //测试类
    public static class TestClass{
        //ID
        private int id;
        //名字
        private String name;
        //年龄
        private int age;

        public int getId() {
            return id;
        }

        public void setId(int id) {
            this.id = id;
        }

        public String getName() {
            return name;
        }

        public void setName(String name) {
            this.name = name;
        }

        public int getAge() {
            return age;
        }

        public void setAge(int age) {
            this.age = age;
        }
    }

}

结果:
在这里插入图片描述
本文参考至:JAVA“动态”为类添加属性

  • 8
    点赞
  • 47
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
Java中,类的属性通常是在类定义时声明的,但是如果需要动态给类添加属性,可以使用Java反射机制来实现。具体步骤如下: 1. 获取该类的Class对象,可以使用Class.forName("类的全限定名")方法或者类的.class属性获取。 2. 使用Class类的getDeclaredFields()方法获取该类的所有属性。 3. 使用Class类的getDeclaredMethod()方法获取该类的set和get方法。 4. 使用Field类的setAccessible(true)方法设置为可访问(因为有些属性可能是私有的)。 5. 使用Field类的set()方法给该类的实例对象设置属性值。 示例代码如下: ``` public class Test { public static void main(String[] args) throws Exception { // 获取类的Class对象 Class<?> clazz = Class.forName("com.example.Person"); // 获取该类的所有属性 Field[] fields = clazz.getDeclaredFields(); // 遍历属性添加属性 Field newField = new Field("newField", String.class); Field[] newFields = new Field[fields.length + 1]; System.arraycopy(fields, 0, newFields, 0, fields.length); newFields[fields.length] = newField; // 定义set和get方法 Method setNewField = clazz.getDeclaredMethod("setNewField", String.class); Method getNewField = clazz.getDeclaredMethod("getNewField"); // 创建对象并设置新属性值 Object obj = clazz.newInstance(); newField.setAccessible(true); newField.set(obj, "newFieldValue"); // 调用set方法设置新属性值 setNewField.invoke(obj, "newFieldValue"); // 调用get方法获取新属性值 System.out.println(getNewField.invoke(obj)); } } class Person { private String name; private int age; public void setNewField(String value) { this.newField = value; } public String getNewField() { return this.newField; } } ```

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值