Java 反射基础

1、什么是反射

反射 (Reflection) 是 Java 的特征之一,它容许运行中的 Java 程序获取自身的信息,而且能够操做类或对象的内部属性。java

总而言之,经过反射,咱们能够在运行时得到程序或程序集中每个类型的成员和成员的信息。程序中通常的对象的类型都是在编译期就肯定下来的,而 Java 反射机制能够动态地建立对象并调用其属性,这样的对象的类型在编译期是未知的。mysql

反射的核心是 JVM 在运行时才动态加载类或调用方法/访问属性,它不须要事先(写代码的时候或编译期)知道运行对象是谁。sql

1.1 反射机制主要提供如下功能

  • 运行时判断任意一个对象所属的类。
  • 运行时构造任意一个类的对象。
  • 运行时判断任意一个类所具备的成员变量和方法。
  • 运行时调用任意一个对象的方法

当咱们的程序在运行时,须要动态的加载一些类,这些类可能以前用不到因此不用加载到jvm,而是在运行时根据须要才加载。数据库

例如咱们的项目底层有时是用mysql,有时用oracle,须要动态地根据实际状况加载驱动类,这个时候反射就有用了,假设 com.java.dbtest.mySqlConnection,com.java.dbtest.oracleConnection这两个类咱们要用,这时候咱们的程序就写得比较动态化,经过Class tc = Class.forName("com.java.dbtest.mySqlConnection");经过类的全类名让jvm在服务器中找到并加载这个类,而若是是oracle则传入的参数就变成另外一个了。这时候就能够看到反射的好处了,这个动态性就体现出java的特性了!数组

2、反射的主要用途

不少人都认为反射在实际的 Java 开发应用中并不普遍,其实否则。当咱们在使用 IDE时,咱们输入一个对象或类并想调用它的属性或方法时,一按点号,编译器就会自动列出它的属性或方法,这里就会用到反射。bash

反射最重要的用途就是开发各类通用框架。不少框架(好比 Spring)都是配置化的(好比经过 XML 文件配置 Bean),为了保证框架的通用性,它们可能须要根据配置文件加载不一样的对象或类,调用不一样的方法,这个时候就必须用到反射,运行时动态加载须要加载的对象。服务器

3、反射的基本运用

上面咱们提到了反射能够用于判断任意对象所属的类,得到Class对象,构造任意一个对象以及调用一个对象。oracle

3.1 Java 的反射机制的实现借助于4个类:class,Constructor,Field,Method;

其中class表明的是类对象,Constructor-类的构造器对象,Field-类的属性对象,Method-类的方法对象,经过这四个对象咱们能够粗略的看到一个类的各个组成部分。其中最核心的就是Class类,它是实现反射的基础,它包含的方法咱们在第一部分已经进行了基本的阐述。应用反射时咱们最关心的通常是一个类的构造器、属性和方法,下面咱们主要介绍Class类中针对这三个元素的方法:框架

3.2 得到Class对象

Class类的实例表示Java应用运行时的类(class and enum)或接口(interface and annotation)(每一个Java类运行时都在JVM里表现为一个Class对象,可经过类名.class,类型.getClass(),Class.forName("类名")等方法获取Class对象)。基本类型boolean,byte,char,short,int,long,float,double和关键字void一样表现为Class对象。jvm

声明普通的Class对象,在编译器并不会检查Class对象的确切类型是否符合要求,若是存在错误只有在运行时才得以暴露出来。可是经过泛型声明指明类型的Class对象,编译器在编译期将对带泛型的类进行额外的类型检查,确保在编译期就能保证类型的正确性。

使用 Class 类的 forName 静态方法:

public static Class<?> forName(String className)

好比在 JDBC 开发中经常使用此方法加载数据库驱动:
Class.forName(driver);
复制代码

直接获取某一个对象的 class

Class<?> klass = int.class;
Class<?> classInt = Integer.TYPE;

复制代码

调用某个对象的 getClass() 方法

StringBuilder str = new StringBuilder("123");
Class<?> klass = str.getClass();
复制代码

3.3 判断是否为某个类的实例

通常地,咱们用 instanceof 关键字来判断是否为某个类的实例。同时咱们也能够借助反射中 Class 对象的 isInstance() 方法来判断是否为某个类的实例,它是一个 native 方法:

public native boolean isInstance(Object obj);
复制代码

3.4 建立实例

使用Class对象的newInstance()方法来建立Class对象对应类的实例

classObj.newInstance() 只可以调用public类型的无参构造函数,此方法是过期的

Class<?> c = String.class;
Object str = c.newInstance();
复制代码

先经过Class对象获取指定的Constructor对象,再调用Constructor对象的newInstance()方法来建立实例。

能够根据传入的参数,调用任意构造函数,在特定状况下,能够调用私有的构造函数,此方法是推荐使用的

//获取String所对应的Class对象
Class<?> c = String.class;
//获取String类带一个String参数的构造器
Constructor constructor = c.getConstructor(String.class);
//根据构造器建立实例
Object obj = constructor.newInstance("23333");
System.out.println(obj);
复制代码

3.5 获取方法

getMethods()

返回某个类的全部public方法,包括本身声明和从父类继承的

getDeclaredMethods()

获取全部本类本身的方法,不问访问权限,不包括从父类继承的方法

getMethod(String name, Class<?>... parameterTypes)

方法返回一个特定的方法,其中第一个参数为方法名称,后面的参数为方法的参数对应Class的对象。

Method getDeclaredMethod(String name, Class<?>... params)

方法返回一个特定的方法,其中第一个参数为方法名称,后面的参数为方法的参数对应Class的对象

操做私有方法

/**
 * 访问对象的私有方法
 * 为简洁代码,在方法上抛出总的异常,实际开发别这样
 */
private static void getPrivateMethod() throws Exception{
    //1. 获取 Class 类实例
    TestClass testClass = new TestClass();
    Class mClass = testClass.getClass();

    //2. 获取私有方法
    //第一个参数为要获取的私有方法的名称
    //第二个为要获取方法的参数的类型,参数为 Class...,没有参数就是null
    //方法参数也可这么写 :new Class[]{String.class , int.class}
    Method privateMethod =
            mClass.getDeclaredMethod("privateMethod", String.class, int.class);

    //3. 开始操做方法
    if (privateMethod != null) {
        //获取私有方法的访问权
        //只是获取访问权,并非修改实际权限
        privateMethod.setAccessible(true);

        //使用 invoke 反射调用私有方法
        //privateMethod 是获取到的私有方法
        //testClass 要操做的对象
        //后面两个参数传实参
        privateMethod.invoke(testClass, "Java Reflect ", 666);
    }
}

复制代码

3.6 获取构造函数

Constructor[] getConstructors()

得到类的全部公共构造函数

Constructor getConstructor(Class[] params)

得到使用特殊的参数类型的公共构造函数,

Constructor[] getDeclaredConstructors()

得到类的全部构造函数

Constructor getDeclaredConstructor(Class[] params)

得到使用特定参数类型的构造函数

3.7 获取成员变量字段

Field[] getFields()

得到类的全部公共字段

Field getField(String name)

得到命名的公共字段

Field[] getDeclaredFields()

得到类声明的全部字段

Field getDeclaredField(String name)

得到类声明的命名的字段

修改私有变量

**
 * 修改对象私有变量的值
 * 为简洁代码,在方法上抛出总的异常
 */
private static void modifyPrivateFiled() throws Exception {
    //1. 获取 Class 类实例
    TestClass testClass = new TestClass();
    Class mClass = testClass.getClass();

    //2. 获取私有变量
    Field privateField = mClass.getDeclaredField("MSG");

    //3. 操做私有变量
    if (privateField != null) {
        //获取私有变量的访问权
        privateField.setAccessible(true);

        //修改私有变量,并输出以测试
        System.out.println("Before Modify:MSG = " + testClass.getMsg());

        //调用 set(object , value) 修改变量的值
        //privateField 是获取到的私有变量
        //testClass 要操做的对象
        //"Modified" 为要修改为的值
        privateField.set(testClass, "Modified");
        System.out.println("After Modify:MSG = " + testClass.getMsg());
    }
}
复制代码

3.8 调用方法

当咱们从类中获取了一个方法后,咱们就能够用 invoke() 方法来调用这个方法。invoke 方法的原型为:

public Object invoke(Object obj, Object... args)
        throws IllegalAccessException, IllegalArgumentException,
           InvocationTargetException
复制代码

下面是一个实例

public class test1 {
    public static void main(String[] args) throws IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {
        Class<?> klass = methodClass.class;
        //建立methodClass的实例
        Object obj = klass.newInstance();
        //获取methodClass类的add方法
        Method method = klass.getMethod("add",int.class,int.class);
        //调用method对应的方法 => add(1,4)
        Object result = method.invoke(obj,1,4);
        System.out.println(result);
    }
}
class methodClass {
    public final int fuck = 3;
    public int add(int a,int b) {
        return a+b;
    }
    public int sub(int a,int b) {
        return a+b;
    }
}
复制代码

3.9 利用反射建立数组

数组在Java里是比较特殊的一种类型,它能够赋值给一个Object Reference。下面咱们看一看利用反射建立数组的例子:

public static void testArray() throws ClassNotFoundException {
        Class<?> cls = Class.forName("java.lang.String");
        Object array = Array.newInstance(cls,25);
        //往数组里添加内容
        Array.set(array,0,"hello");
        Array.set(array,1,"Java");
        Array.set(array,2,"fuck");
        Array.set(array,3,"Scala");
        Array.set(array,4,"Clojure");
        //获取某一项的内容
        System.out.println(Array.get(array,3));
    }
复制代码

其中的Array类为java.lang.reflect.Array类。咱们经过Array.newInstance()建立数组对象,它的原型是:

public static Object newInstance(Class<?> componentType, int length)
        throws NegativeArraySizeException {
        return newArray(componentType, length);
    }
复制代码

而 newArray 方法是一个 native 方法

private static native Object newArray(Class<?> componentType, int length)
        throws NegativeArraySizeException;
复制代码

3.10 反射修改常量值

判断可不能够修改常量
相关文章
相关标签/搜索