Java基础 - 反射(2)

news2025/4/18 20:24:47

文章目录

  • 示例
    • 5. 通过反射获得类的private、 protected、 默认访问修饰符的属性值。
    • 6. 通过反射获得类的private方法。
    • 7. 通过反射实现一个工具BeanUtils, 可以将一个对象属性相同的值赋值给另一个对象

接上篇:

示例

5. 通过反射获得类的private、 protected、 默认访问修饰符的属性值。

import java.lang.reflect.Field;

class Student {
    private String privateField = "私有属性值";
    protected String protectedField = "受保护属性值";
    String defaultField = "默认访问属性值"; // 包级私有
    public String publicField = "公有属性值";
}
public class ReflectionExample5 {
    public static void main(String[] args) {
        Student student = new Student();
        try {
            // 获取类的 Class 对象
            Class<?> clazz = Student.class;
            // ================== 访问私有属性 ==================
            Field privateField = clazz.getDeclaredField("privateField");
            privateField.setAccessible(true); // 解除私有访问限制
            String privateValue = (String) privateField.get(student);
            System.out.println("privateField: " + privateValue);
            // ================== 访问受保护属性 ==================
            Field protectedField = clazz.getDeclaredField("protectedField");
            protectedField.setAccessible(true); // 无需继承关系即可访问
            String protectedValue = (String) protectedField.get(student);
            System.out.println("protectedField: " + protectedValue);
            // ================== 访问默认(包级私有)属性 ==================
            Field defaultField = clazz.getDeclaredField("defaultField");
            defaultField.setAccessible(true);
            String defaultValue = (String) defaultField.get(student);
            System.out.println("defaultField: " + defaultValue);
            // ================== 访问公有属性 ==================
            // 方式 1:getDeclaredField + setAccessible(强制访问)
            Field publicField1 = clazz.getDeclaredField("publicField");
            publicField1.setAccessible(true); // 即使公有也强制解除限制(非必须)
            String publicValue1 = (String) publicField1.get(student);
            System.out.println("publicField(强制访问): " + publicValue1);
            // 方式 2:直接通过 getField 获取(不推荐,仅用于对比)
            Field publicField2 = clazz.getField("publicField");
            String publicValue2 = (String) publicField2.get(student);
            System.out.println("publicField(正常访问): " + publicValue2);
        } catch (NoSuchFieldException e) {
            System.err.println("字段不存在: " + e.getMessage());
        } catch (IllegalAccessException e) {
            System.err.println("访问权限失败: " + e.getMessage());
        }
    }
}

关键操作说明

  1. getDeclaredField() 方法
    - 作用:获取类中声明的所有字段(包括 private/protected/默认/public)
    - 需要手动调用 setAccessible(true) 来突破非 public 属性的访问限制。
  2. setAccessible(true)
    - 方法:Field.setAccessible(true)
    - 意义:解除 Java 的访问控制检查,允许操作非公有字段。
    - 安全性警告:此操作会绕过 Java 的封装机制,仅建议在框架等需要深度操作时使用。
  3. 字段值与对象实例的关联
    - 对于 实例字段:必须传入具体对象实例(如 student),通过 field.get(object) 获取值。
    - 对于 静态字段:可直接传入 null,如 field.get(null)。

6. 通过反射获得类的private方法。

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

class MyClass3 {
    private String privateMethod(String param) {
        return "私有方法被调用,参数: " + param;
    }
}
public class ReflectionExample6 {
    public static void main(String[] args) {
        try {
            // 创建类的实例
            MyClass3 myObject = new MyClass3();
            // 获取Class对象
            Class<?> clazz = MyClass3.class;
            // 获取私有方法,需指定方法名和参数类型
            Method method = clazz.getDeclaredMethod("privateMethod", String.class);
            // 解除访问限制(关键步骤)
            method.setAccessible(true);
            // 调用方法,传入实例及参数
            String result = (String) method.invoke(myObject, "Hello");
            // 输出结果
            System.out.println("调用结果: " + result);
        } catch (NoSuchMethodException e) {
            System.err.println("方法未找到: " + e.getMessage());
        } catch (IllegalAccessException e) {
            System.err.println("非法访问: " + e.getMessage());
        } catch (InvocationTargetException e) {
            System.err.println("方法内部错误: " + e.getCause());
        }
    }
}

7. 通过反射实现一个工具BeanUtils, 可以将一个对象属性相同的值赋值给另一个对象

import java.lang.reflect.Field;
import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;

class BeanUtils {
    /**
     * 将源对象的属性值复制到目标对象(浅拷贝)
     * @param source 源对象
     * @param target 目标对象
     */
    public static void copyProperties(Object source, Object target) {
        if (source == null || target == null) {
            throw new IllegalArgumentException("源对象和目标对象不能为 null");
        }

        // 获取源对象和目标对象的所有字段(包含父类字段)
        List<Field> sourceFields = getAllFields(source.getClass());
        List<Field> targetFields = getAllFields(target.getClass());

        for (Field sourceField : sourceFields) {
            // 查找目标对象中与源对象字段同名的字段
            Field targetField = findField(targetFields, sourceField.getName());
            if (targetField == null) continue;

            // 检查类型是否兼容(支持基本类型和包装类型)
            if (!isTypeCompatible(sourceField.getType(), targetField.getType())) continue;

            try {
                // 设置字段可访问性
                sourceField.setAccessible(true);
                targetField.setAccessible(true);

                // 复制值
                Object value = sourceField.get(source);
                targetField.set(target, value);
            } catch (IllegalAccessException e) {
                // 处理无法访问的异常
                System.err.println("字段复制失败: " + sourceField.getName() + " -> " + targetField.getName());
            }
        }
    }

    /**
     * 获取类及其父类的所有字段(非静态)
     */
    private static List<Field> getAllFields(Class<?> clazz) {
        List<Field> fields = new java.util.ArrayList<>();
        while (clazz != null && clazz != Object.class) {
            fields.addAll(Arrays.stream(clazz.getDeclaredFields())
                    .filter(f -> !isStatic(f))
                    .collect(Collectors.toList()));
            clazz = clazz.getSuperclass();
        }
        return fields;
    }

    /**
     * 根据字段名从字段列表中查找字段
     */
    private static Field findField(List<Field> fields, String fieldName) {
        return fields.stream()
                .filter(f -> f.getName().equals(fieldName))
                .findFirst()
                .orElse(null);
    }

    /**
     * 判断字段是否为静态
     */
    private static boolean isStatic(Field field) {
        return java.lang.reflect.Modifier.isStatic(field.getModifiers());
    }

    /**
     * 判断源类型与目标类型是否兼容
     */
    private static boolean isTypeCompatible(Class<?> sourceType, Class<?> targetType) {
        // 处理基本类型与包装类型的兼容(例如 int -> Integer)
        if (sourceType.isPrimitive()) {
            return targetType.isPrimitive() ? sourceType.equals(targetType) : getWrapperType(sourceType).equals(targetType);
        } else if (targetType.isPrimitive()) {
            return getWrapperType(targetType).equals(sourceType);
        } else {
            return targetType.isAssignableFrom(sourceType);
        }
    }

    /**
     * 获取基本类型对应的包装类型
     */
    private static Class<?> getWrapperType(Class<?> primitiveType) {
        if (primitiveType == int.class) return Integer.class;
        if (primitiveType == long.class) return Long.class;
        if (primitiveType == boolean.class) return Boolean.class;
        if (primitiveType == byte.class) return Byte.class;
        if (primitiveType == char.class) return Character.class;
        if (primitiveType == short.class) return Short.class;
        if (primitiveType == double.class) return Double.class;
        if (primitiveType == float.class) return Float.class;
        return primitiveType;
    }
}

// 父类
class Person {
    protected String name;
    private int age;
    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;
    }
}

// 子类
class Employee extends Person {
    private double salary;
    private Integer departmentId;

    public double getSalary() {
        return salary;
    }

    public void setSalary(double salary) {
        this.salary = salary;
    }

    public Integer getDepartmentId() {
        return departmentId;
    }

    public void setDepartmentId(Integer departmentId) {
        this.departmentId = departmentId;
    }
}

// 目标类
class EmployeeDTO {
    private String name;
    private int age;        // 基本类型
    private Double salary;  // 包装类型
    private Integer departmentId;

    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;
    }

    public Double getSalary() {
        return salary;
    }

    public void setSalary(Double salary) {
        this.salary = salary;
    }

    public Integer getDepartmentId() {
        return departmentId;
    }

    public void setDepartmentId(Integer departmentId) {
        this.departmentId = departmentId;
    }
}

public class ReflectionExample7 {
    public static void main(String[] args) {
        // 准备数据
        Employee emp = new Employee();
        emp.name = "张三";
        emp.setAge(30);
        emp.setSalary(15000.5);
        emp.setDepartmentId(101);
        // 目标对象
        EmployeeDTO dto = new EmployeeDTO();
        // 复制属性
        BeanUtils.copyProperties(emp, dto);
        // 验证结果
        System.out.println("DTO.name: " + dto.getName());
        System.out.println("DTO.age: " + dto.getAge());
        System.out.println("DTO.salary: " + dto.getSalary());
        System.out.println("DTO.departmentId: " + dto.getDepartmentId());
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2332382.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

P1162 洛谷 填涂颜色

题目描述 思考 看数据量 30 而且是个二维的&#xff0c;很像走迷宫 直接深搜&#xff01; 而且这个就是搜连通块 代码 一开始的15分代码&#xff0c;想的很简单&#xff0c;对dfs进行分类&#xff0c;如果是在边界上&#xff0c;就直接递归&#xff0c;不让其赋值&#xff0c…

docker安装nginx,基础命令,目录结构,配置文件结构

Nginx简介 Nginx是一款轻量级的Web服务器(动静分离)/反向代理服务器及电子邮件&#xff08;IMAP/POP3&#xff09;代理服务器。其特点是占有内存少&#xff0c;并发能力强. &#x1f517;官网 docker安装Nginx &#x1f433; 一、前提条件 • 已安装 Docker&#xff08;dock…

用Django和AJAX创建一个待办事项应用

用Django和AJAX创建一个待办事项应用 推荐超级课程: 本地离线DeepSeek AI方案部署实战教程【完全版】Docker快速入门到精通Kubernetes入门到大师通关课AWS云服务快速入门实战目录 用Django和AJAX创建一个待办事项应用让我们创建一个简单的 Django 项目,其中包含不同类型的 A…

JavaScript:游戏开发的利器

在近年来的科技迅速发展中&#xff0c;JavaScript 已逐渐成为游戏开发领域中最受欢迎的编程语言之一。它的跨平台特性、广泛的社区支持、丰富的库和框架使得开发者能够快速、有效地创建各种类型的游戏。本文将深入探讨 JavaScript 在游戏开发中的优势。 一、跨平台支持 JavaSc…

C语言今天开始了学习

好多年没有弄了&#xff0c;还是捡起来弄下吧 用的vscode 建议大家参考这个配置 c语言vscode配置 c语言这个语言简单&#xff0c;但是今天听到了一个消息说python 不知道怎么debug。人才真多啊

电商素材革命:影刀RPA魔法指令3.0驱动批量去水印,实现秒级素材净化

本文 去除水印实操视频展示电商图片水印处理的困境​影刀 RPA 魔法指令 3.0 强势登场​利用魔法指令3.0两步实现去除水印操作关于影刀RPA 去除水印实操视频展示 我们这里选择了4张小红书里面比较帅气的图片&#xff0c;但凡用过小红书的都知道&#xff0c;小红书右下角是会有小…

CVA6:支持 Linux 的 RISC-V CPU CORE-V

RISC-V 是一种开源的可扩展指令集架构 (ISA)&#xff0c;在过去几年中广受欢迎。RISC-V 的主要特性之一是它采用整体架构中性设计&#xff0c;支持浮点运算、加载存储架构、符号扩展加速和多路复用器简化。这使得 RISC-V 成为 FPGA 上软处理器的经济实惠的选择。自 RISC-V ISA …

轻奢宅家|约克VRF带你畅享舒适居家体验

下班回到家最期待什么&#xff1f;当然是一阵阵沁人心脾的舒适感扑面而来啦&#xff01;      想要从头到脚都舒服自在&#xff1f;答案就在眼前——就是它&#xff01;约克VRF中央空调&#xff01;      约克VRF中央空调独特的臻静降噪技术&#xff0c;让空调运行音轻…

uniapp微信小程序图片生成水印

整体思路&#xff1a; 用户通过uni.chooseImage选择图片后&#xff0c;获得图片文件的path和size。通过path调用uni.getImageInfo获取图片信息&#xff0c;也就是图片宽高。图片宽高等比缩放至指定大小&#xff0c;不然手机处理起来非常久&#xff0c;因为手机随便拍拍就很大。…

不用额外下载jar包,idea快速查看使用的组件源码

以nacos为例子&#xff0c;在idea中引入了nacos依赖&#xff0c;就可以查看源码了。 2. idea选择open&#xff08;不关闭项目直接选择file-open也可以&#xff09;, 在maven的仓库里找到对应的包&#xff0c;打开 2.idea中选择 jar包&#xff0c;选择 add as library 3.这样j…

网络通讯协议UDP转发TCP工具_UdpToTcpRelay_双向版

UDP/TCP网络转发器程序说明书 1. 程序概述 本程序是一个高性能网络数据转发工具&#xff0c;支持UDP和TCP协议之间的双向数据转发&#xff0c;并具备以下核心功能&#xff1a; 协议转换&#xff1a;实现UDP↔TCP协议转换数据转换&#xff1a;支持十六进制/ASCII格式的数据转…

DIA——边缘检测

1.边缘 边缘是像素的突变位置。 2.常见边缘检测算法 通过找到一阶导数的极值点或者二阶导数的过零点来确定边缘像素的位置。边缘检测通常使用算子&#xff0c;即特定的卷积核。通过差分对离散的像素点求导&#xff0c;然后转化成卷积核进行卷积。使用卷积统一涵盖求导&…

【万象论坛】论坛系统测试报告

一、项目背景 1.1项目起因 在当今数字化浪潮下&#xff0c;互联网技术呈爆发式发展&#xff0c;新技术、新框架、新应用场景不断涌现。从大型企业的数字化转型到初创公司的技术创新&#xff0c;各个层面都离不开互联网技术的支撑。然而&#xff0c;技术人员在学习与工作过程中…

【AI工具】FastGPT:开启高效智能问答新征程

前言 在人工智能飞速发展的当下&#xff0c;各类 AI 工具如雨后春笋般涌现。FastGPT 作为一款基于大语言模型&#xff08;LLM&#xff09;的知识图谱问答系统&#xff0c;凭借其强大的数据处理和模型调校能力&#xff0c;为用户带来了便捷的使用体验。今天&#xff0c;就让我们…

华为数字芯片机考2025合集1已校正

单选 1&#xff0e;以下低功耗措施中&#xff0c;哪种不是降低电路翻转率的方法? A.在不进行算术运算的时候&#xff0c;使这些模块的输入保持不变&#xff0c;不让新的操作数进来 B.采用Gray 码或One‐hot 码作为状态机编码 C.减少电路中的glitch D.重新安排“if‐else”表达…

HackMyVM - todd记录

HackMyVM - toddhttps://mp.weixin.qq.com/s/E_-hepdfY-0veilL1fl2QA

【完整可用】使用openhtmltopdf生成PDF(带SVG)

文章目录 前言OpenHTMLToPDF 简介maven配置依赖字体文件demo代码其他资源放置截图防止maven编译字体文件 前言 AI和网上都是跑不起来或者版本过低的&#xff0c;还有各种BUG的。本文都是查阅官方文档得出的。如果你能跑起来请给个大大的赞&#xff01; OpenHTMLToPDF 简介 Ope…

CTF web入门之爆破

爆破 web21: 打开burp进行抓包 通过对密码进行解析。得知密码是由拼接而来 admin:1 选择要攻击的参数 攻击方式。 选择payload方式 。。添加参数 1&#xff0c;2&#xff0c;3。账号 分隔符 密码 选择加密方式。添加buse64.去掉url字符。不然buse64后&#xff0c;会在u…

7-openwrt-one通过web页面配置访客网络、无线中继等功能

前几个章节一直在介绍编译、分区之类的,都还没正常开始使用这个路由器的wifi。默认wifi是没有启动的,前面还是通过手动修改uci配置启动的,这个章节介绍下官方web页面的使用。特别是访客网络、无线中继 1、开启wifi,配置wifi基本信息 我们使用有线连接路由器,通过192.168.…

Android使用声网SDK实现音视频互动(RTC)功能

一、前期准备 1、注册声网账号 声网官网 2、创建项目 拿到AppID&#xff0c;主要证书 二、代码部分 先上一下官方提供的demo地址&#xff1a; Agora-RTC-QuickStart: 此仓库包含 Agora RTC Native SDK 的QuickStart示例项目。 - Gitee.comhttps://gitee.com/agoraio-comm…