32 java反射机制详解

简介: 1 反射快速上手案例

4.png


1 反射快速上手案例


public class ReflectionQuestion {
    public static void main(String[] args) throws IOException, ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException {
//        创建properties对象
        Properties properties = new Properties();
//        加载文件
        properties.load(new FileReader("E:\\Codes\\myProject\\JiShiTongXun\\reflectionDemo\\re.properties"));
//        获取值
        String classpaht = properties.getProperty("classpaht");
        String method = properties.getProperty("method");
        System.out.println(classpaht);
//        使用反射机制创建类
//        ①加载类
        Class aClass = Class.forName(classpaht);
//        ②通过得到的aClass对象,可以得到加载的类(配置文件中的类)的实例
        ReflectionDemo rfd = (ReflectionDemo) aClass.newInstance();
//        ③得到方法的对象
        Method method1 = aClass.getMethod(method);
//        ④通过方法对象,调用方法
        method1.invoke(rfd);
    }
}


2 使用反射的步骤


第一步:载Class类,会返回一个Class类型的对象

//①加载Class类,会返回一个Class类型的对象
        Class aClass = Class.forName("加载类全路径");

第二步:通过Class类对象,得到需要加载的类Dog类的对象实例

//①加载Class类,会返回一个Class类型的对象
            Class aClass = Class.forName(classpahttset);

第三步:通过aClass得到加载类的Method对象

// ③通过aClass得到加载类(Dog)的Method对象
        Method sayColor = aClass.getMethod("方法名");

第四步:通过Method对象,调用方法

//④通过Method对象,调用方法
        sayColor.invoke(aClass);


3 反射详解:


17.png

反射相关的主要类:

(1) java.lang.Class:代表一个类,Class对象表示某个类加载后在堆中的对象

(2) java.lang.reflect.Method:代表类的方法,Method对象表示某个类的方法

(3) java.lang.reflect.Field:代表类的成员变量

(4) java.lang.reflect.Constructor:代表类的构造方法;


4 反射的优化


Method、Field、Constructor都有setAccessible()方法

setAccessible方法是为了启动和禁止访问安全检查的开关

参数为true,表示反射的对象在访问时取消检查,提高反射效率。参数为false时,表示执行访问检查。

//        优化反射
        sayColor.setAccessible(true);


5 Class类特点梳理


45.png

(1)Class也是一个类,它的父类也是Object

(2)Class类对象不是new出来的,而是系统创建的

(3)对于某个类的Class对象,在内存中只能存在一份,因为类只加载一次;

(4)每个类的实例都会知道自己是由哪个Class类生成

(5)通过Class类对象可以得到一个类的完整结构;

(6)Class对象存放在堆上

(7)类的字节码二进制文件是存放在方法区的


6 Class类的常用方法


public class ClassDemo {
    public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchFieldException {
        Class aClass = Class.forName("com.zinksl.reflectionDemo.reflection01.Dog");
        //        获取包名
        String packgname = aClass.getPackage().getName();
        System.out.println(packgname);
        //        获取全类名
        System.out.println(aClass.getName());
        //        通过Class对象,创建需求类对象示例
        Dog dog = (Dog)aClass.newInstance();
        //        通过反射获取属性
        Field color = aClass.getField("color");
        String colors = (String) color.get(dog);
        System.out.println(colors);
        //        通过反射给属性设置值
        color.set(dog,"黄色");
        //        通过反射获取所有属性
        Field[] fields = aClass.getFields();
    }
}


7 获取Class类的方式


7.1 在编译阶段获取Class类对象,通过Class.forName()方法


如果已知全类名,则可以通过forName()方法获取

应用场景:用于配置文件,读取全类名,路径加载类


7.2 在类加载阶段可以通过,类名.class获取


应用场景:多用于参数传递,如:通过反射得到对于构造器对象;


7.3 在运行阶段可以通过,对象名.getClass()方法获取


7.4 通过类加载器获取Class对象


8 java中哪些类有Class对象


(1)成员内部类、静态内部类、局部内部类、匿名内部类

(2)Interface:接口

(3)数组

(4)枚举

(5)annotation:注解

(6)基本数据类型

(7)void


9 类加载


【反射是java实现动态语言的关键,也是通过反射实现动态加载的】

类加载可分为静态加载和动态加载

静态加载,在编译期就加载类,如果找不到就报错,依赖性比较强

动态加载,在运行时加载需要的类,如果运行时用不到该类,就算找不到也不报错,依赖性比较弱


10 通反射获取类的结构信息:


方法名 说明
getName 获取全类名
getSimpleName 获取简单类名
getFields 获取所有public修饰的属性,包括本类及父类
getDeclardeFields 获取本类所有属性
getMethods 获取所有public修饰的方法,包括本类及父类
getDeclardeMethods 获取本类所有方法
getConstructors 获取所有public修饰的构造器,包括本类
getDeclardeConstructors 获取本类所有构造器
getPackage 以Package形式返回包信息
getSuperClass 以Class形式返回父类信息
getInterfaces 以Class[]形式返回接口信息
getAnnotations 以Annotation[] 形式返回注解信息

代码演示

public class ReflectionTest {
    public static void main(String[] args) {
    }
    @Test
    public void api() throws ClassNotFoundException, InstantiationException, IllegalAccessException {
//        getMethods  获取所有public修饰的方法,包括本类及父类
//        getDeclardeMethods  获取本类所有方法
//        getConstructors 获取所有public修饰的构造器,包括本类及父类
//        getDeclardeConstructors 获取本类所有构造器
//        getPackage  以Package形式返回包信息
//        getSuperClass 以Class形式返回父类信息
//        getInterfaces 以Class[]形式返回接口信息
//        getAnnotations  以Annotation[] 形式返回注解信息
//        1首先得到对应的Class对象
        Class<?> aClass = Class.forName("com.zinksl.reflectionDemo.reflectionTest.Person");
//        2 得到目标类对象
        Person person = (Person) aClass.newInstance();
//        getName 获取全类名
//        getSimpleName 获取简单类名
        String name = aClass.getName();
        System.out.println("全类名:"+name);
        System.out.println("简单类名:"+aClass.getSimpleName());
//        getFields 获取所有public修饰的属性,包括本类及父类
//        getDeclardeFields 获取本类所有属性
        Field[] fields = aClass.getFields();
        for (Field f:fields){
            System.out.println("本类及父类的所有共有属性:"+f.getName());
        }
        Field[] declaredFields = aClass.getDeclaredFields();
        for (Field declaredField : declaredFields) {
            System.out.println("本类所有属性:"+declaredField.getName());
        }
    }
}
class Person{
    public String name;
    public int age;
    private String hoby;
    public Person() {
    }
    public Person(String name, int age, String hoby) {
        this.name = name;
        this.age = age;
        this.hoby = hoby;
    }
    public void sayHello(){
        System.out.println("hello");
    }
    private void sayHi(){
        System.out.println("hi");
    }
    /**
     * 获取
     * @return name
     */
    public String getName() {
        return name;
    }
    /**
     * 设置
     * @param name
     */
    public void setName(String name) {
        this.name = name;
    }
    /**
     * 获取
     * @return age
     */
    public int getAge() {
        return age;
    }
    /**
     * 设置
     * @param age
     */
    public void setAge(int age) {
        this.age = age;
    }
    /**
     * 获取
     * @return hoby
     */
    public String getHoby() {
        return hoby;
    }
    /**
     * 设置
     * @param hoby
     */
    public void setHoby(String hoby) {
        this.hoby = hoby;
    }
    public String toString() {
        return "Person{name = " + name + ", age = " + age + ", hoby = " + hoby + "}";
    }
}


11 java.lang.reflect.Field类


(1)getModifiers:以int形式返回修饰符

【说明:默认修饰符:0,public:1,private:2,

protected:4,static:8,final:16;public static:1+8】

(2)getType:以Class形式返回类型

(3)getName:返回方法名


12 java.lang.reflect.Method类


(1)getModifiers:以int形式返回修饰符

【说明:默认修饰符:0,public:1,private:2,

protected:4,static:8,final:16;public static:1+8】

(2)getReturnType:以Class形式返回类型

(3)getName:返回方法名

(4)getParameterType:以Class[] 返回参数类型数组


13 java.lang.reflect.Constructor类


(1)getModifiers:以int形式返回修饰符

【说明:默认修饰符:0,public:1,private:2,

protected:4,static:8,final:16;public static:1+8】

(4)getParameterType:以Class[] 返回参数类型数组

(3)getName:返回构造器名(全类名)


13 通过反射创建对象


(1) 调用类中被public修饰的无参构造


(2)调用类中指定的构造器


(3)Class类相关方法


①newInstance :调用类中无参构造器,获取对应类对象

②getConstructor(Class…):根据参数列表获取对应的构造器对象

③getDecalaredConstructor(Class…):根据参数列表获取对应的构造器对象


(4)Constructor类相关方法


①setAccessible;爆破

②newInstance(Object…):调用构造器

代码示例:

public class ReflectGetObject {
    public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException {
//        1 先获取Person类的Class对象
        Class personClass = Class.forName("com.zinksl.reflectionDemo.reflectGetObject.Person");
//        2通过public无参构造器,创造实例
        Object o = personClass.newInstance();
        System.out.println(o);
//        3通过public有参构造器,创造实例
        //①先得到构造器
        Constructor constructor = personClass.getConstructor(String.class);
        //②通过构造器创建实例
        Object o1 = constructor.newInstance("马云");
        System.out.println(o1);
//        4 通过私有构造器,创造实例
        //先得到构造器
        Constructor constructor1 = personClass.getDeclaredConstructor(String.class,int.class,String.class);
//        爆破
        constructor1.setAccessible(true);//使用反射,创建私有化构造器,对象
        Object o2 = constructor1.newInstance("大黄", 80, "溜圈");
        System.out.println(o2);
    }
}
class Person{
    private String name ="码云";
    private int age = 20;
    public String hoby;
    public Person() {
    }
    public Person(String name){
        this.name = name;
    }
    private Person(String name, int age, String hoby) {
        this.name = name;
        this.age = age;
        this.hoby = hoby;
    }
    /**
     * 获取
     * @return name
     */
    public String getName() {
        return name;
    }
    /**
     * 设置
     * @param name
     */
    public void setName(String name) {
        this.name = name;
    }
    /**
     * 获取
     * @return age
     */
    public int getAge() {
        return age;
    }
    /**
     * 设置
     * @param age
     */
    public void setAge(int age) {
        this.age = age;
    }
    /**
     * 获取
     * @return hoby
     */
    public String getHoby() {
        return hoby;
    }
    /**
     * 设置
     * @param hoby
     */
    public void setHoby(String hoby) {
        this.hoby = hoby;
    }
    public String toString() {
        return "Person{name = " + name + ", age = " + age + ", hoby = " + hoby + "}";
    }
}

14 通过反射访问类中成员


(1) 根据属性名 获取Field对象


Field f = Class对象.getDeclaredField(属性名);


(2)爆破 f.setAccessible(true); f是 Field的对象;


(3)访问


f.set(o,值);o表示对象

syso(f.get(o)); o表示对象


(4)如果是静态属性,则set、get中的o属性,可以设置为null;


代码示例:

public class ReflectionGetShuxing {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, NoSuchFieldException, IllegalAccessException, InvocationTargetException, InstantiationException {
//1获得Class类对象
        Class<?> aClass = Class.forName("com.zinksl.reflectionDemo.reflectGetObject.Person2");
//        2创建对象
        Object o = aClass.newInstance();
//        3 获取Field的对象
        Field hoby = aClass.getField("hoby");
//        4 设置属性
        hoby.set(o,"抽烟喝酒烫头");
        System.out.println(o);
        Field age = aClass.getDeclaredField("age");
        age.setAccessible(true);
        age.set(o,9999);
        System.out.println(o);
    }
}
class Person2{
    private static String name ="码云";
    private int age = 20;
    public String hoby;
    public Person2() {
    }
    public Person2(String name){
        this.name = name;
    }
    private Person2(String name, int age, String hoby) {
        this.name = name;
        this.age = age;
        this.hoby = hoby;
    }
    /**
     * 获取
     * @return name
     */
    public String getName() {
        return name;
    }
    /**
     * 设置
     * @param name
     */
    public void setName(String name) {
        this.name = name;
    }
    /**
     * 获取
     * @return age
     */
    public int getAge() {
        return age;
    }
    /**
     * 设置
     * @param age
     */
    public void setAge(int age) {
        this.age = age;
    }
    /**
     * 获取
     * @return hoby
     */
    public String getHoby() {
        return hoby;
    }
    /**
     * 设置
     * @param hoby
     */
    public void setHoby(String hoby) {
        this.hoby = hoby;
    }
    public String toString() {
        return "Person{name = " + name + ", age = " + age + ", hoby = " + hoby + "}";
    }
}


目录
相关文章
|
7天前
|
安全 Java 数据库连接
Java中的异常处理机制及其最佳实践
【8月更文挑战第12天】在Java的世界里,异常处理是维护程序健壮性的关键环节之一。它不仅涉及到错误检测,还关乎如何优雅地恢复或报告错误。本文将深入探讨Java的异常处理机制,包括异常类型、捕获异常的策略以及异常的最佳处理实践。我们将一起学习如何通过有效的异常管理来提升代码质量和用户体验。
|
7天前
|
Java 数据库连接 开发者
Java中的异常处理机制深度解析
【8月更文挑战第13天】本文旨在深入探讨Java编程语言中一个至关重要的组成部分——异常处理机制。我们将从基本概念入手,逐步展开讨论异常处理在Java语言设计中的角色和重要性,以及如何正确利用这一机制来提高代码的健壮性和可维护性。文章将通过分析异常处理的最佳实践,揭示如何在复杂的应用程序中有效地管理和处理错误情况。
|
15天前
|
安全 Java 调度
解锁Java并发编程高阶技能:深入剖析无锁CAS机制、揭秘魔法类Unsafe、精通原子包Atomic,打造高效并发应用
【8月更文挑战第4天】在Java并发编程中,无锁编程以高性能和低延迟应对高并发挑战。核心在于无锁CAS(Compare-And-Swap)机制,它基于硬件支持,确保原子性更新;Unsafe类提供底层内存操作,实现CAS;原子包java.util.concurrent.atomic封装了CAS操作,简化并发编程。通过`AtomicInteger`示例,展现了线程安全的自增操作,突显了这些技术在构建高效并发程序中的关键作用。
41 1
|
15天前
|
存储 Oracle 安全
揭秘Java并发核心:深入Hotspot源码腹地,彻底剖析Synchronized关键字的锁机制与实现奥秘!
【8月更文挑战第4天】在Java并发世界里,`Synchronized`如同导航明灯,确保多线程环境下的代码安全执行。它通过修饰方法或代码块实现独占访问。在Hotspot JVM中,`Synchronized`依靠对象监视器(Object Monitor)机制实现,利用对象头的Mark Word管理锁状态。
27 1
|
3天前
|
存储 SQL 关系型数据库
深入MySQL锁机制:原理、死锁解决及Java防范技巧
深入MySQL锁机制:原理、死锁解决及Java防范技巧
|
6天前
|
Java 程序员 编译器
深入浅出Java异常处理机制
在Java编程的世界中,异常处理就像是我们生活中的急救包,它帮助我们处理程序运行时出现的意外情况。本文将带你了解Java异常处理的基础知识,探索异常类型,学习如何捕获和处理它们,并讨论最佳实践。让我们一起走进Java异常处理的世界,学会如何使用这个强大的工具来保护我们的程序。
|
7天前
|
安全 Java 程序员
Java中的异常处理机制:从基础到高级
【8月更文挑战第13天】在Java编程中,异常处理是一个不可或缺的部分。本文将深入探讨Java的异常处理机制,从基本概念出发,逐步解析try-catch-finally语句块的使用,探讨自定义异常类的设计,以及介绍高级异常处理技术如try-with-resources和异常链。通过这些内容的学习,读者将能够更加有效地管理和控制程序运行时的错误。
|
9天前
|
消息中间件 负载均衡 Java
"深入Kafka核心:探索高效灵活的Consumer机制,以Java示例展示数据流的优雅消费之道"
【8月更文挑战第10天】在大数据领域,Apache Kafka凭借其出色的性能成为消息传递与流处理的首选工具。Kafka Consumer作为关键组件,负责优雅地从集群中提取并处理数据。它支持消息的负载均衡与容错,通过Consumer Group实现消息的水平扩展。下面通过一个Java示例展示如何启动Consumer并消费数据,同时体现了Kafka Consumer设计的灵活性与高效性,使其成为复杂消费场景的理想选择。
37 4
|
9天前
|
消息中间件 负载均衡 Java
"Kafka核心机制揭秘:深入探索Producer的高效数据发布策略与Java实战应用"
【8月更文挑战第10天】Apache Kafka作为顶级分布式流处理平台,其Producer组件是数据高效发布的引擎。Producer遵循高吞吐、低延迟等设计原则,采用分批发送、异步处理及数据压缩等技术提升性能。它支持按消息键值分区,确保数据有序并实现负载均衡;提供多种确认机制保证可靠性;具备失败重试功能确保消息最终送达。Java示例展示了基本配置与消息发送流程,体现了Producer的强大与灵活性。
28 3
|
12天前
|
Java 开发者
深入解析Java中的异常处理机制
在Java的世界中,异常处理是维护程序健壮性的基石之一。本文将通过实例演示和理论分析相结合的方式,探讨Java异常处理机制的工作原理及其最佳实践。我们将从异常的基本概念出发,逐步深入到异常类的层次结构、捕获异常的策略以及自定义异常的使用场景,旨在为读者提供一个全面而深入的视角来理解和应用Java中的异常处理。
11 2