深入分析java中的反射机制

简介: 对于java中的反射机制,面试的时候也是会经常的提问到,在网上看了很多文章也查了很多资料,于是花了一部分时间整理了一下,也算是查漏补缺吧。

一、反射概念


在正式讲解反射之前,为了很好的去理解它我们先从一个案例说起。请看下面的代码:

public class User {
    private String name;
    private int age;
    public User(){}
    public User(String name, int age) {
        this.name = name;
        this.age = age;
    }
    public void test(){
        System.out.println("年龄是:"+name);
    }
}

这是一个最简单不过的类,当我们使用的时候直接new出来一个User对象即可。因为这个类是我们自己定义的,所以在使用的时候我们知道User有两个字段name和age,还有无参和有参构造方法,另外的test方法我们也可以直接调用(因为其是public)。


现在出现一个问题,如果这个user类不是我们自己定义的,我们从外部看不到里面有什么东西,而且我们又想去知道内部长什么样,比如说有几个字段、方法、构造方法、共有还是私有的等等,这时候该怎么办呢?这时候java语言在设计的时候为我们提供了一个机制,就是反射机制。他能够很方便的去解决我们的问题。


通过上面的例子我相信你也能对java反射机制的功能了解一二了,现在我们再来整理一下:

java反射机制允许我们程序员在 程序运行的时候获取一个类的各种内部信息,比如说modifiers(诸如public, static 等)、superclass(例如Object)、实现之interfaces(例如Cloneable),也包括fields和methods的所有信息。 更重要的是我们还能够修改这些信息

下面我们就来好好看一下,java中的反射机制是如何在运行时获取这些类的内部信息的。


二、深入分析java反射机制


1、获取Class类


在java中万事万物皆对象,User user=new User()一行代码我们知道了user是User类的实例对象,通过Student stu=new Student()我们知道了stu是Student的实例对象,但是我们想过没,User和Student又是谁的对象呢?没错就是Class类的实例对象。那这个Class类是什么东西,内部长什么样子呢?这时候我们很自然的联想到使用反射机制。使用反射机制就可以获取到这个class。


这里有三种方式可以获取这个Class,我们来看一下代码:

//这里假设我们之前不知道User类的内部状态
public class Test {
    public static void main(String[] args) {
        User user = new User();
        // 第一种表示方式:通过类名
        Class c1 = User.class;
        // 第二中表达方式:通过对象
        Class c2 = user.getClass();
        //第三种表达方式
        try {
            Class c3 = Class.forName("com.fdd.reflecttest.User");
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

上面的c1、c2、c3都是Class类的实例,表示的都是User类。

当然,不仅仅是User这些类,对于基本数据类型甚至是包括void我们也可以使用这个方法。

Class c1 = int.class;
Class c2 = String.class;
Class c3 = double.class;
Class c4 = Double.class;
Class c5 = void.class;

现在就可以直接创建User类的实例了。

User user = (User)c1.newInstance();
//需要有无参数的构造方法

现在我们对反射机制中获取Class类的方法进行一个总计

v2-653423aa9f7f3fa1a9dbdbd13ad11ef4_1440w.jpg

2、获取类的方法


现在通过反射看一下User类内部的样子,打印一下(把这个操作封装在了一个方法中):

//打印类的信息,包括类的成员函数、成员变量(只获取成员函数)
    public static void printClassMethodMessage(Object obj){
        //1、首先要获取类的类类型
        Class c = obj.getClass();
        //2、获取类的名称
        System.out.println("类的名称是:"+c.getName());
        //3、获取方法信息
        Method[] ms = c.getMethods();
        for(int i = 0; i < ms.length;i++){
            //第一步:得到方法的返回值类型的类类型
            Class returnType = ms[i].getReturnType();
            System.out.print(returnType.getName()+" ");
            //第二步:得到方法的名称
            System.out.print(ms[i].getName()+"(");
            //第三步:获取方法参数类型--->得到的是参数列表的类型的类类型
            Class[] paramTypes = ms[i].getParameterTypes();
            for (Class class1 : paramTypes) {
                System.out.print(class1.getName()+",");
            }
            System.out.println(")");
        }
    }

下面我们把我们的User类传进去,打印一下。

public class Test {
    public static void main(String[] args) {
        User user=new User("18",20);
        ClassUtil.printClassMethodMessage(user);
    }
}
//output
//类的名称是:com.fdd.reflecttest.User
//void test()
//void wait()
//void wait(long,int,)
//void wait(long,)
//boolean equals(java.lang.Object,)
//java.lang.String toString()
//int hashCode()
//java.lang.Class getClass()
//void notify()
//void notifyAll()

v2-9f78f0452d0da3f4a3d171f82ece7033_1440w.jpg

3、获取类的属性


(1)获取所有属性

public static void printFieldMessage(Object obj) {
        Class c = obj.getClass();
        //Field类封装了关于成员变量的操作
        //  getFields()方法获取的是所有的public的成员变量的信息
        //  getDeclaredFields获取的是该类自己声明的成员变量的信息
        //Field[] fs = c.getFields();
        Field[] fs = c.getDeclaredFields();
        for (Field field : fs) {
            //1、获取字段类型
            Class fieldType = field.getType();
            String typeName = fieldType.getName();
            //2、获取字段名称
            String fieldName = field.getName();
            //打印字段类型和字段名称
            System.out.println(typeName+" "+fieldName);
        }
    }

上面有两种获取属性的方法。重点是for循环。我们来测试一下

public class Test {
    public static void main(String[] args) {
        User user=new User("18",20);
        ClassUtil.printFieldMessage(user);
    }
}
//output
//java.lang.String name
//int age

直接就会输出我们的字段类型和名称。

(2)获取指定属性

在这里我们的User类中name、age字段增加getter和setter方法

public static Object printFieldMsgBySelf(Object obj) {
        Class c = obj.getClass();
        try {
            //1、获取指定字段
            Field age_field = c.getDeclaredField("age");
            //2、实例化一个User类
            Object object=c.newInstance();
            //3、使用反射机制打破封装
            age_field.setAccessible(true);
            //4、给我们实例化的对象重新设置年龄
            age_field.set(object, 100);
            //5、返回这个object
            return object;
        }  catch (Exception e) {
            e.printStackTrace();
        }
        return c;
}

然后我们测试一下

public class Test {
    public static void main(String[] args) {
        User userBefore=new User("18",20);
        System.out.println(userBefore.getAge());
        User userAfter=(User) ClassUtil.printFieldMsgBySelf(userBefore);
        System.out.println(userAfter.getAge());
    }
}
//output
//20
//100

在这里,我们在printFieldMsgBySelf方法中通过反射重新设置了age年龄的值,输出之后已成功更改。

v2-19e1d5b76da12466f6d0ebb6aabef389_1440w.jpg

4、获取类的构造方法


public static void printConMessage(Object obj){
        Class c = obj.getClass();
        //Constructor中封装了构造函数的信息
        //  (1)getConstructors获取所有的public的构造函数
        //  (2)getDeclaredConstructors得到所有的构造函数
        //第一步:获取构造函数
        Constructor[] cs = c.getDeclaredConstructors();
        for (Constructor constructor : cs) {
            //获取构造函数名字
            System.out.print(constructor.getName()+"(");
            //获取构造函数的参数列表
            Class[] paramTypes = constructor.getParameterTypes();
            for (Class class1 : paramTypes) {
                System.out.print(class1.getName()+",");
            }
            System.out.println(")");
        }
}

然后我们同样的测试一下

public class Test {
    public static void main(String[] args) {
        User user=new User("18",20);
        ClassUtil.printConMessage(user);
    }
}
//output
//com.fdd.reflecttest.User()
//com.fdd.reflecttest.User(java.lang.String,int,)

跟我们之前的构造方法一样。

v2-c009b3c7d1108881aaa0fa5c5660c5a4_1440w.jpg

5、获取User类的父类和接口


我们在这里定义一个Human类(里面什么也没有),然后定义一个UserInterface接口,让User继承它就好了。

//取得父类和接口
public static void getUserSuperClassAndInterface(Object obj) {
        Class c = obj.getClass();
        //取得父类:父类只能有一个
        Object superClass = c.getSuperclass();
        System.out.println(superClass);
        //取得接口:接口可以有很多
        Object[] superInterface = c.getInterfaces();
        for (Object myInter:superInterface) {
            System.out.println(myInter);
        }
}
//调用这个方法后的输出结果:
//class com.fdd.reflecttest.Human
//interface com.fdd.reflecttest.UserInterface

然后我们在Test中去测试一下就可以了,测试方法很简单,我们只需要调用这个方法就可以。


小结:在上面的案例中,我们使用反射机制能够获取类的方法、字段、构造方法、父类和接口,当然也可以获取一些其他的信息。但是这里有一点重要的知识,那就是我们不仅可以获取上面的这些信息,还可以修改它,这对一个类来说是极其的不安全的。这一点我们需要注意。


下面我们就来看看反射到底用什么用途。


三、使用反射机制


1、通过反射了解泛型的本质


java中集合的泛型是防止错误输入的;只在编译阶段有效,只要绕过编译就无效啦。比如下面的代码:

ArrayList list1=new ArrayList();
ArrayList<String> list2=new ArrayList<String>();

对于list1来说我们可以添加任何对象,但是对于list2来说,就必须输入String类型对象,这就是泛型作用。(当然这里只是简单的提一下,泛型还有很多知识)。在运行时候,是不区分输入什么的。

下面我们就来验证一下。

public class Test {
    public static void main(String[] args) throws Exception {
        ArrayList list1=new ArrayList();
        ArrayList<String> list2=new ArrayList<String>();
        Class c1=list1.getClass();
        Class c2=list2.getClass();
        //在运行期间泛型无效,所以c1和c2应该是一样的。
        System.out.print(c1==c2);//true
        //由于泛型失效,所以此时list当然可以添加任何对象
        Method m=c2.getMethod("add",Object.class);
        m.invoke(list2,20);//向list2集合中添加一个int 型的值;绕过编译
    }
}
//output
//true

上面的输出已经说明一切。


2、spring中使用


学习Spring的时候,我们知道Spring主要有Ioc和AOP两大思想,它利用的是反射机制,依赖注入就不用多说了,而对于Spring的核心AOP来说,使用了动态代理,其实底层也是反射。


当然还有动态代理模式、web拦截器等等,使用极其广泛。但是这里有一个窍门需要我们注意一下,那就是面试的时候,使用反射机制往往能实现违反java语言设计原则的事,比如说String类型是不可变类型的,我们使用反射机制就可以使他变成可变类型的。

OK,反射原理基本上先到这里,对于其用途,在相应的文章中会提到,这里算是反射机制的基础知识吧,因为最终是要去用的。

相关文章
|
16天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
45 2
|
20天前
|
Java 编译器
探索Java中的异常处理机制
【10月更文挑战第35天】在Java的世界中,异常是程序运行过程中不可避免的一部分。本文将通过通俗易懂的语言和生动的比喻,带你了解Java中的异常处理机制,包括异常的类型、如何捕获和处理异常,以及如何在代码中有效地利用异常处理来提升程序的健壮性。让我们一起走进Java的异常世界,学习如何优雅地面对和解决问题吧!
|
1月前
|
XML 安全 Java
Java反射机制:解锁代码的无限可能
Java 反射(Reflection)是Java 的特征之一,它允许程序在运行时动态地访问和操作类的信息,包括类的属性、方法和构造函数。 反射机制能够使程序具备更大的灵活性和扩展性
41 5
Java反射机制:解锁代码的无限可能
|
19天前
|
Java 数据库连接 开发者
Java中的异常处理机制及其最佳实践####
在本文中,我们将探讨Java编程语言中的异常处理机制。通过深入分析try-catch语句、throws关键字以及自定义异常的创建与使用,我们旨在揭示如何有效地管理和响应程序运行中的错误和异常情况。此外,本文还将讨论一些最佳实践,以帮助开发者编写更加健壮和易于维护的代码。 ####
|
25天前
|
安全 IDE Java
Java反射Reflect机制详解
Java反射(Reflection)机制是Java语言的重要特性之一,允许程序在运行时动态地获取类的信息,并对类进行操作,如创建实例、调用方法、访问字段等。反射机制极大地提高了Java程序的灵活性和动态性,但也带来了性能和安全方面的挑战。本文将详细介绍Java反射机制的基本概念、常用操作、应用场景以及其优缺点。 ## 基本概念 ### 什么是反射 反射是一种在程序运行时动态获取类的信息,并对类进行操作的机制。通过反射,程序可以在运行时获得类的字段、方法、构造函数等信息,并可以动态调用方法、创建实例和访问字段。 ### 反射的核心类 Java反射机制主要由以下几个类和接口组成,这些类
57 2
|
29天前
|
存储 Java 关系型数据库
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践,包括连接创建、分配、复用和释放等操作,并通过电商应用实例展示了如何选择合适的连接池库(如HikariCP)和配置参数,实现高效、稳定的数据库连接管理。
52 2
|
1月前
|
存储 缓存 安全
🌟Java零基础:深入解析Java序列化机制
【10月更文挑战第20天】本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
25 3
|
1月前
|
Java 关系型数据库 数据库
面向对象设计原则在Java中的实现与案例分析
【10月更文挑战第25天】本文通过Java语言的具体实现和案例分析,详细介绍了面向对象设计的五大核心原则:单一职责原则、开闭原则、里氏替换原则、接口隔离原则和依赖倒置原则。这些原则帮助开发者构建更加灵活、可维护和可扩展的系统,不仅适用于Java,也适用于其他面向对象编程语言。
22 2
|
1月前
|
安全 Java UED
深入理解Java中的异常处理机制
【10月更文挑战第25天】在编程世界中,错误和意外是不可避免的。Java作为一种广泛使用的编程语言,其异常处理机制是确保程序健壮性和可靠性的关键。本文通过浅显易懂的语言和实际示例,引导读者了解Java异常处理的基本概念、分类以及如何有效地使用try-catch-finally语句来处理异常情况。我们将从一个简单的例子开始,逐步深入到异常处理的最佳实践,旨在帮助初学者和有经验的开发者更好地掌握这一重要技能。
27 2
|
1月前
|
Java 数据库连接 开发者
Java中的异常处理机制####
本文深入探讨了Java语言中异常处理的核心概念,通过实例解析了try-catch语句的工作原理,并讨论了finally块和throws关键字的使用场景。我们将了解如何在Java程序中有效地管理错误,提高代码的健壮性和可维护性。 ####