不是吧,还有人连Java最强大的技术之一:反射还没搞懂?赶紧码住

相信很多人都知道反射可以说是Java中最强大的技术了,它可以做的事情太多太多,很多优秀的开源框架都是通过反射完成的,比如最初的很多注解框架,后来因为java反射影响性能,所以被运行时注解APT替代了,java反射有个开源框架jOOR相信很多人都用过,不过我们还是要学习反射的基础语法,这样才能自己写出优秀的框架,当然这里所讲的反射技术,是学习Android插件化技术、Hook技术等必不可少的!

概述

Java 反射是可以让我们在运行时获取类的方法、属性、父类、接口等类的内部信息的机制。也就是说,反射本质上是一个“反着来”的过程。我们通过new创建一个类的实例时,实际上是由Java虚拟机根据这个类的Class对象在运行时构建出来的,而反射是通过一个类的Class对象来获取它的定义信息,从而我们可以访问到它的属性、方法,知道这个类的父类、实现了哪些接口等信息。

何为反射?

Java的 反射机制 是在运行状态中,对于任意一个类,都能够 知道这个类的所有属性和方法 ;对于任意一个对象,都能够 调用它的任意一个方法和属性 ;这种 动态获取的信息以及动态调用对象的方法的功能 称为 Java 语言的反射机制

简而言之,只要你给我一个 .class ——类的名字,我就能通过反射获取到类的属性和方法。

反射是很多高级技术的基础,Java 中的注解、动态代理,各种框架注入 Spring 、 MyBatis 等都用到了反射技术。

反射机制能做什么

 1. 在运行时判断一个对象所属的类
 2. 在运行时可以构造任意一个类的对象
 3. 在运行时可以任意判断一个类所包含的成员变量和方法
 4. 在运行时可以任意调用一个对象的方法
 5. 生成动态代理

 Class类

我们知道使用javac能够将.java文件编译为.class文件,这个.class文件包含了我们对类的原始定义信息(父类、接口、构造器、属性、方法等)。.class文件在运行时会被ClassLoader加载到Java虚拟机(JVM)中,当一个.class文件被加载后,JVM会为之生成一个Class对象,我们在程序中通过new实例化的对象实际上是在运行时根据相应的Class对象构造出来的。确切的说,这个Class对象实际上是java.lang.Class泛型类的一个实例,比如Class对象即为一个封装了MyClass类的定义信息的Class实例。由于java.lang.Class类不存在公有构造器,因此我们不能直接实例化这个类,我们可以通过以下方法获取一个Class对象。
 

Class 类在 JDK 中的定义

public final class Class
extends Object
implements Serializable, GenericDeclaration, Type, AnnotatedElement
复制代码

类 Class 的实例表示运行中的 Java 应用程序中的类和接口。

枚举 是一种类,注释 是一种接口。

每个 数组 也属于一个类,这个类反映为一个 类对象 ,由具有相同元素类型和维数的所有数组共享。

原始Java类型(布尔型、字节型、char型、short型、int型、long型、float型和double型)和关键字void也被表示为类对象。

类的加载过程

Class类的实例表示正在运行的Java应用程序中的类和接口。每个类只会产生一个Class对象,在类加载的时候自动创建 

不是吧,还有人连Java最强大的技术之一:反射还没搞懂?赶紧码住_第1张图片

获取Class类的三种方式

1 通过 class.forname()来获取Class对象    

        Class clazz = Class.forName("com.mashibing.entity.Emp");
        System.out.println(clazz.getPackage());
        System.out.println(clazz.getName());
        System.out.println(clazz.getSimpleName());
        System.out.println(clazz.getCanonicalName());

2 通过类名.class来获取Class对象

        Class clazz = Emp.class;
        System.out.println(clazz.getPackage());
        System.out.println(clazz.getName());
        System.out.println(clazz.getSimpleName());
        System.out.println(clazz.getCanonicalName());

3 通过对象的getClass()来获取Class对象

        Class clazz = new Emp().getClass();
        System.out.println(clazz.getPackage());
        System.out.println(clazz.getName());
        System.out.println(clazz.getSimpleName());
        System.out.println(clazz.getCanonicalName());

4如果是一个基本数据类型,那么可以通过Type的方式来获取Class对象

        Class type = Integer.TYPE;
        
        System.out.println(type.getName());
        System.out.println(type.getCanonicalName());

反射常用到的API

  • 获取类的构造方法:

    public Constructor getConstructor(Class... parameterTypes)
                                  throws NoSuchMethodException,
                                         SecurityException
    
    public Constructor[] getConstructors()
                                     throws SecurityException
    复制代码
  • 获取类的成员变量

    public Field getField(String name)
                   throws NoSuchFieldException,
                          SecurityException
    
    public Field[] getFields()
                      throws SecurityException
    复制代码
  • 获取类的方法

    public Method getMethod(String name,
                            Class... parameterTypes)
                     throws NoSuchMethodException,
                            SecurityException
    
    public Method getMethod(String name,
                            Class... parameterTypes)
                     throws NoSuchMethodException,
                            SecurityException

反射在 Spring 中的应用举例

反射在众多框架中都有普遍的应用。比如 Spring IOC 容器帮我们实例化众多的bean,下面我们简单模拟一下 反射 在其中起到的作用。

此处使用的案例接这篇:【设计模式】代理模式那些事儿:静态代理,动态代理,JDK的动态代理,cglib,Spring AOP

Spring 配置文件:


复制代码

使用的时候直接这样就能拿到定义的类了:

ApplicationContext ctx = new ClassPathXmlApplicationContext("app_aop.xml");
Pony pony = (Pony) ctx.getBean("pony");
复制代码

那么是怎么做到的呢?就是通过 反射

Spring 通过配置文件实例化对象,并将其放到容器的过程大概就是(模拟):

//伪代码
//1.解析元素的id属性得到该字符串值为“pony”  
String idStr = "pony";  
//解析元素的class属性得到该字符串值为“com.xblzer.dp.proxy.springaop.Pony”  
String classStr = "com.xblzer.dp.proxy.springaop.Pony";  
//利用反射机制,通过classStr获取Class类对象  
Class cls = Class.forName(classStr);  
//实例化对象  
Object obj = cls.newInstance();  
//放到Spring容器  
Map container = new HashMap<>();
container.put(idStr, obj); 

你可能感兴趣的:(java,开发语言)