Java反射机制调用对象的方法 —— 将一个对象的属性值赋值给另一个对象的属性

本文介绍如何使用Java反射机制解决对象属性批量复制的问题,特别是在处理不同类型的对象时,避免重复编写循环和判断代码。通过创建通用工具类,实现源对象属性到目标对象的自动映射和赋值,适用于各种场景。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

通过反射机制调用对象的某个方法 -- 将一个对象的属性值赋值给另一个对象的属性

模拟一个场景:

众所周知,EasyExcel导出Excel文档是依赖于注解完成的,在实体类需要导出的属性上面加上注解,导出的时候会自动识别该属性。

假如我们现在需要导出用户的信息,又不想污染原本的实体类,又要过滤掉password这个属性。那么我们可以另外创建一个实体类,不包含password属性。然后我们将查到的用户信息,赋值给新创建的对象即可。

赋值这一步是比较繁琐的,我们需要写循环,然后判断,然后赋值。等将来需要导出管理员信息的时候我们又要写循环,然后判断,然后赋值、、、

怎么能写一个共用的方法去帮我们做循环、判断、赋值这些事呢?源对象类型不确定,目标对象类型不确定。

于是,反射机制来了。它来了,它来了,它哼着小曲走来了、、、

通过 Java 的反射机制,程序员可以更深入地控制程序的运行过程。例如,在程序运行时由用户输入一个类名,然后动态获取该类拥有的构造、属性和方法,甚至调用任意类的任意方法。

Java 反射机制主要提供了以下功能,这些功能都位于 java.lang.reflect包下。

  • 在运行时判断任意一个对象所属的类。
  • 在运行时构造任意一个类的对象。
  • 在运行时判断任意一个类所具有的成员变量和方法。
  • 在运行时调用任意一个对象的方法。
  • 生成动态代理。

开发环境

请参照: 基于SpringBoot构建分模块项目

创建对象UserOne、UserTwo

UserOne为与数据库表对应的实体类,UserTwo为即将要通过EasyExcel导出的对象

package com.wayne.common.entity;

/**
 * @author Wayne
 * @date 2019/6/5
 */
public class UserOne {

    private Integer id;

    private String username;

    private String password;

   // Getter and Setter 、、、
}
package com.wayne.common.dto;

/**
 * @author Wayne
 * @date 2019/6/5
 */
public class UserTwo {

    private Integer id;

    private String username;

    // Getter and Setter 、、、
}

编写基于反射的工具类

通过反射创建对象、调用方法

package com.wayne.common.utils;

import com.google.common.collect.Lists;
import com.wayne.common.exception.CopyPropertyException;

import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.List;

/**
 * @author Wayne
 * @date 2019/6/5
 */
public class BaseUtil {

    /**
     * 将一个集合中对象的值拷贝到另一个对象,属性相同即赋值
     * @param source 源数据,将此对象数据取出
     * @param tClass 目标对象,将取出的数据赋值到该对象中
     * @param <T> 源数据类型
     * @param <E> 目标数据类型
     * @return 被赋值后的目标对象集合
     * @throws CopyPropertyException 自定义异常,通过反射创建对象或调用方法时抛出的异常
     */
    public static <T, E> List<E> copyProperties(List<T> source, Class<E> tClass) throws CopyPropertyException {

        // 判断传入源数据是否为空,如果空,则抛自定义异常
        if(null == source) {
            throw new CopyPropertyException("数据源为空");
        }

        // 创建一个集合,用于存储目标对象,全部数据抓换完成后,将该集合返回
        List<E> targetList = Lists.newArrayList();

        // 循环取到单个源对象
        for(T t : source) {
            // 获取源对象的类的详情信息
            Class<?> sClass = t.getClass();
            // 获取源对象的所有属性
            Field[] sFields = sClass.getDeclaredFields();
            // 获取目标对象的所有属性
            Field[] tFields = tClass.getDeclaredFields();

            E target;
            try {
                // 通过类的详情信息,创建目标对象 这一步等同于UserTwo target = new UserTwo();
                target = tClass.newInstance();
            } catch (Exception e) {
                e.printStackTrace();
                throw new CopyPropertyException("目标对象创建失败");
            }

            // 循环取到源对象的单个属性
            for(Field sField : sFields) {
                // 循环取到目标对象的单个属性
                for(Field tField : tFields) {

                    // 判断源对象的属性名、属性类型是否和目标对象的属性名、属性类型一致
                    if(sField.getName().equals(tField.getName()) && sField.getGenericType().equals(tField.getGenericType())) {

                        try {
                            // 获取源对象的属性名,将属性名首字母大写,拼接如:getUsername、getId的字符串
                            String sName = sField.getName();
                            char[] sChars = sName.toCharArray();
                            sChars[0] -= 32;
                            String sMethodName = "get" + String.valueOf(sChars);
                            // 获得属性的get方法
                            Method sMethod = sClass.getMethod(sMethodName);
                            // 调用get方法
                            Object sFieldValue = sMethod.invoke(t);

                            // 获取目标对象的属性名,将属性名首字母大写,拼接如:setUsername、setId的字符串
                            String tName = tField.getName();
                            char[] tChars = tName.toCharArray();
                            tChars[0] -= 32;
                            String tMethodName = "set" + String.valueOf(tChars);
                            // 获得属性的set方法
                            Method tMethod = tClass.getMethod(tMethodName, tField.getType());
                            // 调用方法,并将源对象get方法返回值作为参数传入
                            tMethod.invoke(target, sFieldValue);

                            break;
                        } catch (Exception e) {
                            e.printStackTrace();
                            throw new CopyPropertyException("转换失败,请检查属性类型是否匹配");
                        }
                    }
                }
            }
            // 将通过反射创建的目标对象放入集合中
            targetList.add(target);
        }
        // 返回集合
        return targetList;
    }
}

测试结果

测试结果

预留占位

开发怎能不留扩展字段 (¬_¬)…

### C# 中检查对象属性存在性及获取属性值 对于C#而言,在处理对象及其属性时,可以利用反射实现这一需求。具体来说,`Type` 类提供了 `GetProperty` 方法用于检索指定名称的公共属性[^1]。 ```csharp using System; using System.Reflection; public class ExampleClass { public string PropertyName { get; set; } } // 假设 obj 是目标实例化后的对象, propertyName 是要查找的属性名字符串表示形式. PropertyInfo propertyInfo = obj.GetType().GetProperty(propertyName); if (propertyInfo != null && propertyInfo.CanRead) { object value = propertyInfo.GetValue(obj); // 获取属性值 } ``` 上述代码片段展示了如何通过反射技术判断某特定命名的属性是否存在于给定的对象内,并尝试读取该属性的具体数值。当 `GetProperty` 返回非空结果时表示找到了匹配项;而进一步确认 `CanRead` 属性确保此字段确实支持读操作之后,则可通过调用 `GetValue()` 来取得实际存储的数据。 ### Java 中检查对象属性存在性及获取属性值Java环境下,同样可以通过反射机制完成相似的任务。这里展示了一个简单的例子说明怎样验证并访问成员变量: ```java import java.lang.reflect.Field; public class User { private String name; // getter and setter... } try { Class<?> clazz = userInstance.getClass(); Field field = clazz.getDeclaredField("name"); boolean isAccessibleBefore = field.canAccess(userInstance); field.setAccessible(true); Object fieldValue = field.get(userInstance); // 取得属性值 } catch (NoSuchFieldException | IllegalAccessException e) { // Handle exceptions here } ``` 这段程序首先定位到所需的字段定义(`getDeclaredField`),接着调整其可见度设置以便能够跨越封装界限直接存取私有成员(`setAccessible`). 如果一切顺利的话,最后一步就是运用 `get` 函数提取出当前保存于选定位置中的内容了[^2]. ### Vue.js 组件内部检测数据属性的存在性和初始状态 Vue框架下,开发者通常关心的是响应式数据的状态管理以及生命周期钩子内的上下文环境。下面的例子解释了如何在一个组件刚初始化但还未完全准备好呈现之前查询某些预配置好的选项是否存在有效赋值: ```javascript export default { data() { return { message: 'Hello' }; }, created() { console.log(this.$options.data.call(this).message !== undefined ? "Message exists." : "No Message."); } }; ``` 在这个场景里,`created` 钩子函数是在实例创建完成后立即触发的一个理想时机去审查那些即将参与视图渲染过程的关键要素——比如这里的 `message`. 此外值得注意的一点是 `$options` 对象包含了原始传入的各种配置参数副本,因此借助它可以帮助我们更方便地接触到未经加工过的源码级设定[^3].
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值