一文了解JAVA反射(超详尽!)

反射是框架设计的灵魂,只有学好了反射,才能设计出好的框架

一.反射的概述

Java反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态调用对象方法的功能就是Java语言的反射机制.简单来说,就是反射可以帮助我们在动态运行的时候,对于任意一个类,可以获得其所有方法,所有的变量(是所有的!包括私有!)

反射的作用

  • 获取某些类的一些变量,调用某些类的私有方法.(举个栗子,例如在Android开发中我们可以用来开启WiFi热点,调用WifiManager中的setWifiApEnabled()方法)
  • 增加代码的灵活性.(很多主流框架都使用了反射技术,像ssm框架都采用两种技术xml做配置文件+反射技术)
而我们要想使用反射,就要对反射的常用对象有个基本了解
  • Class
    Class类的实例表示正在运行的Java应用程序中的类和接口
  • Constructor
    关于类的单个构造方法的信息以及对它的访问权限
  • Field
    Field 提供有关类或接口的单个字段的信息,以及对它的动态访问权限.
  • Method
    Method 提供关于类或接口上单独某个方法的信息

反射在内存中的样子大概长这样
在这里插入图片描述
这是一个简单的Animal类,这个Animal.java文件,想要被执行的话,首先会被编译成Animal.class文件.而Animal.class想要被执行的话,就会被类加载器加载到JVM中执行,JVM就会将它加载到内存,而加载之后,我们的字节码文件Animal.class在内存中也会有一个对象的表示,别忘了Java语言有句话咋说的,“万事万物,皆为对象”.而.class文件在内存中的对象就是Class对象,所以说获得了Class,才能获得它的构造方法,属性以及方法,相对应的 构造方法在 内存中 对应的是Constructor对象, 属性 对应的就是 Field,而方法呢 对应的就是Method,但是 不管我们想要获得构造方法,属性 还是 方法, 前提都是我们先获得Class对象

二.Class类

  • Java中java.lang.Class类用于表示一个类的字节码(.class)文件
  • 如何得到某个class文件对应的Class对象
    已知类和对象的情况下
    类名.class
    对象.getClass() — Object 类提供
    未知类和对象的情况下
    Class.forName(“包名.类名”)

Class类代表某个类的字节码,并提供了加载字节码的方法:forName(“包名.类名”),forName方法用于加载类字节码到内存中,并封装成一个Class对象.

package JavaReflectTest;

public class Animal {
    public String name;
    private int id;
    public Animal(){
        System.out.println("我是无参构造方法");
    }
    public Animal(int id,String name){
        this.setId(id);
        this.setName(name);
        System.out.println("我是有参构造方法");
    }

    public void eat(){
        System.out.println("我是公有方法");
    }
    private void drink(){
        System.out.println("我是私有方法");
    }
    private void play(String name,String sex){
        System.out.println("我是私有带参方法");
    }

    @Override
    public String toString() {
        return "Animal{" +
                "name='" + name + '\'' +
                ", id=" + id +
                '}';
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }
}

package JavaReflectTest;

public class ClassTest {
    /**
     * 获得Class对象
     * 1.通过类名.class
     * 2.对象.getClass()
     * 3.Class.forName();
     */
    public void demo1() throws ClassNotFoundException {
        //1.通过类名.class的方式
        Class class1 = Animal.class;
        //2.通过对象.getClass()的方式
        Animal animal = new Animal();
        Class class2 = animal.getClass();
        //3.Class类forName();获得(推荐)
        Class class3 = Class.forName("JavaReflectTest.Animal");
    }
    public static void main(String[] args){

    }
}

这三种获得Class对象的方法,推荐使用第三种方法,因为我们通常做反射的时候,都是在不知道类的实例的时候进行操作的,注意第三种方法通过Class.forName();来获得Class对象的时候,会抛出一个ClassNotFoundException 没有发现类的异常,如果我们Class.forName(“路径”); 路径写错了或者没有找到这个类的时候,就会抛出这个异常,当然这个异常处理,你可以选择try…catch处理也可以先向上抛,我这里为了方便,就直接向上抛了

三.Constructor类

  • Constructor类的实例对象代表类的一个构造方法
  • 得到某个类所有的构造方法
    Constructor [] constructors = Class.forName(“java.lang.String”).getConstructors();
  • 得到指定的构造方法并调用
    Constructor constructor = Class.forName(“java.lang.String”).getConstructors(String.class);
    String str = (String)constructor.newInstance(“abc”);
  • Class类的newInstance()方法用来调用类的默认构造方法
    String obj = (String)Class.forName(“java.lang.String”).newInstance();
package JavaReflectTest;

import java.lang.reflect.Constructor;

public class ConstructorTest {
    /**
     *获得无参构造方法
     */
    public void demo1() throws Exception {
        //获得类的字节码文件对应的对象:
        Class class1 = Class.forName("JavaReflectTest.Animal");
        Constructor c = class1.getConstructor();
        Animal animal = (Animal)c.newInstance();//相当于 Animal animal = new Animal();
    }

    /**
     *获得有参构造方法
     */
    public void demo2() throws Exception{
        Class class1 = Class.forName("JavaReflectTest.Animal");
        Constructor c = class1.getConstructor(int.class,String.class);
        Animal animal = (Animal)c.newInstance(2,"猴子");//相当于 Animal animal = new Animal(2,"猴子");
        System.out.println(animal.toString());
    }
    public static void main(String[] args) throws Exception {
        ConstructorTest c = new ConstructorTest();
        c.demo1();
        System.out.println("------------");
        c.demo2();
    }
}

运行结果
在这里插入图片描述

四.Field类

  • Field类代表某个类中的一个成员变量,并提供动态的访问权限
  • Field对象的获得
    Field[] fields = c.getFields();//取得所有public属性(包括父类继承)
    Field[] fields = c.getDeclaredFields();//取得所有声明的属性
  • 得到指定的成员变量
    Field name = c.getField(“name”);
    Field name = c.getDeclaredField(“name”);
  • 设置Filed变量是否可以访问
    field.setAccessible(boolean);
    -Field变量值的读取,设置
    field.get(obj);
    filed.set(obj,value);
package JavaReflectTest;

import java.lang.reflect.Field;

public class FieldTest {
    //测试共有属性
    public void demo1() throws Exception{
        //获得Class
        Class class1 = Class.forName("JavaReflectTest.Animal");
        //获得属性:
        Field field = class1.getField("name");
        //操作属性:
        Animal animal = (Animal)class1.getConstructor().newInstance();
        field.set(animal,"老虎");// animal.name = "老虎";
        //获取值
        Object obj = field.get(animal);
        System.out.println(obj);
        System.out.println(animal);//这样会自动调用toString()方法
    }
    //测试私有方法
    public void demo2() throws Exception{
        //获得Class
        Class class1 = Class.forName("JavaReflectTest.Animal");
        //获得私有属性:
        Field field = class1.getDeclaredField("id");
        //操作属性:
        Animal animal = (Animal)class1.getConstructor().newInstance();
        //私有属性,需要设置一个可访问的权限
        field.setAccessible(true);
        field.set(animal,1);
        //获取值:
        Object obj = field.get(animal);
        System.out.println(obj);
        System.out.println(animal);
    }
    public static void main(String[] args) throws Exception {
        FieldTest fieldTest = new FieldTest();
        fieldTest.demo1();
        System.out.println("----------------");
        fieldTest.demo2();
    }
}

效果图
在这里插入图片描述

五.Method类

  • Method类代表某个类中的一个成员方法
  • Method对象的获得
    获得所有方法
    getDeclaredMethods()
    getMethods()
    -获得指定的方法
    getDeclaredMethod(String name,Class<?>…parameterTypes)
    getMethod(String name,Class<?>…parameterTypes)
  • 通过反射执行方法
    invoke(Object obj,Object…args)
package JavaReflectTest;

import java.lang.reflect.Method;

public class MethodTest {
    //测试公有方法
    public void demo1() throws Exception{
        //获取Class对象
        Class class1 = Class.forName("JavaReflectTest.Animal");
        //实例化
        Animal animal = (Animal)class1.getConstructor().newInstance();
        //获得公有方法
        Method method = class1.getMethod("eat");
        //执行该方法
        method.invoke(animal);// 相当于animal.eat();
    }
    //测试私有方法
    public void demo2() throws Exception{
        //获取Class对象
        Class class1 = Class.forName("JavaReflectTest.Animal");
        //实例化
        Animal animal = (Animal)class1.getConstructor().newInstance();
        //获得私有方法
        Method method = class1.getDeclaredMethod("drink");
        //设置私有的访问权限
        method.setAccessible(true);
        //执行该方法
        method.invoke(animal);//相当于animal.drink();
    }
    public void demo3() throws Exception{
        //获取Class对象
        Class class1 = Class.forName("JavaReflectTest.Animal");
        //实例化
        Animal animal = (Animal)class1.getConstructor().newInstance();
        //获得私有带参的方法
        Method method = class1.getDeclaredMethod("play", int.class, String.class);
        //设置私有的访问权限
        method.setAccessible(true);
        //执行该方法
        Object obj = method.invoke(animal,2,"孔雀"); // 这是有返回值的情况 如果没有返回值可以直接写method.invoke(animal,2,"孔雀");
        System.out.println(obj);//打印返回的东西,没有就是null
    }

    //测试私有带参数的方法
    public static void main(String[] args) throws Exception{
        MethodTest methodTest = new MethodTest();
        methodTest.demo1();
        System.out.println("--------------------");
        methodTest.demo2();
        System.out.println("--------------------");
        methodTest.demo3();

    }

}

效果图
在这里插入图片描述
java反射就介绍到这里
我是羽毛,一个不是很正经的正经人
最好的关系就是互相成就,关注,点赞,收藏你们的三连就是羽毛创作的最大动力
注:如果本博客有任何错误和建议,欢迎留言和私信


文章持续更新,可以关注微信公众号:羽毛的技术栈,每天分享互联网新资讯

发布了3 篇原创文章 · 获赞 27 · 访问量 276

猜你喜欢

转载自blog.csdn.net/qq_43527936/article/details/105571740