反射机制【Java】

36 篇文章 0 订阅
29 篇文章 0 订阅


在一些特定的场景中,我们可能会需要获取一些私有的成员变量或方法的信息,但直接在类外调用是无法成功获取到的,因此我们就需要一种机制来获取一些需要的变量或属性,也就是反射机制。

定义

反射属于是java的一种机制,通过这种机制,可以对处于运行状态中的任意一个类获取到它的所有属性或方法,同时可以调用这些属性和方法。这样一种动态获取信息同时动态调用对象方法或属性的功能就是java的反射机制。

代表了类的实体,一般在运行的java应用程序中表示类和接口;

Class类属于是反射机制的起源。原因是对于一个java文件而言,经编译之后就会生成一个.class文件,之后该文件要经过JVM解析,解析得到一个对象java.lang.Class。也就是说,每个java文件最终都会成为Class类对象的一个实例。此时若是我们需要使用或修改这个类的属性或方法,自然就需要使用反射机制来完成,此时这个类也就成为了一个动态的类。

获得Class对象的方式

进行反射的第一步首先是获取当前需要反射的类的类对象,再通过Class对象的方法来实现反射。常见的获取Class对象的方式主要有3种:

  • 通过使用类对象的getClass()方法获取;
  • 使用.class方法获取;
  • 通过使用Class.forName(“类对象的完整路径名”)获取;

下面使用代码来演示具体的使用方法:

package reflect;

/**
 * Created with IntelliJ IDEA.
 * Description:
 * User: 晨曦
 * Date: 2023-05-10
 * Time: 10:25
 */

class Student{
    private String name="liming";

    public int age=20;

    public Student (){
        System.out.println("一个不带参数的构造方法");
    }

    private Student(String name, int age) {
        this.age=age;
        this.name=name;
        System.out.println("name:"+name+"; age:"+age);
    }

    private void study(String name){
        this.name=name;
        System.out.println(name+" i love study");
    }

    public void play(){
        System.out.println("i want to play");
    }

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

public class testDemo1 {
    public static void main(String[] args) {
        //1.通过getClass()获取Class对象
        Student student=new Student();
        Class c1=student.getClass();


        //2.直接通过类名.class获取类对象
        Class c2=Student.class;

        Class c3=null;
        try {
        //forName的参数必须是完整的路径名(包名+类名)
            c3=Class.forName("reflect.Student");
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
        System.out.println(c1);
        System.out.println(c2);
        System.out.println(c3);


        System.out.println(c1.equals(c2));
        System.out.println(c1.equals(c3));
        System.out.println(c3.equals(c2));

    }
}

在这里插入图片描述

由于一个JVM只会解析出一个Class实例,因此获取的c1,c2,c3三个对象实际都是同一个;

三种获取方式各有千秋,第一种通过getClass获取Class对象的方式需要格外创建一个实例;第二种直接通过 类名.class 的方式更加安全,程序的性能要更高;通过 Class 对象的 forName() 静态方法来获取的方式需要明确类的路径名,使用更加广泛;

反射的具体使用

使用上面的Student类具体演示反射的使用方法,主要就是获取到类中一些私有的方法或属性;

package reflect;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

public class ReflectTest {

    /*
     * 通过反射创建对象
     * */
    public static void reflectNewInstance() {
      try {
            //1.获取类对象
            Class<?> c=Class.forName("reflect.Student");
            //2.创建类的实例
            Student student=(Student) c.newInstance();
        }  catch (Exception e) {
            e.printStackTrace();
        }

    }

    /*
     * 通过反射调用私有的构造方法
     * */

    public static void reflectPrivateConstructor() {

        try {
            //1.获取类对象
            Class<?> c = Class.forName("reflect.Student");
            //获得该类中与参数类型相匹配的构造方法
           Constructor<?> constructor = c.getDeclaredConstructor(String.class, int.class);
            //获得该类的所有公有的构造方法
           /*Constructor<?> constructor=c.getConstructor();*/

            //2.修改访问权限(设置为true)
            constructor.setAccessible(true);
            Student student = (Student) constructor.newInstance("lisi",24);     //调用类中对应的方法  name:lisi; age:24

            ClassLoader classLoader=c.getClassLoader();//获得类的加载器
            System.out.println(classLoader);    //sun.misc.Launcher$AppClassLoader@18b4aac2
            String str=c.getName();//获得类的完整路径名
            System.out.println(str);        //reflect.Student
        } catch (Exception e) {
            e.printStackTrace();
        }


    }


/*
* 通过反射获取私有属性
* */
    public static void reflectPrivateField(){

        try {
            //1.获取类对象
            Class<?> c = Class.forName("reflect.Student");
            //2.获得某个属性对象
            Field field=c.getDeclaredField("name");
            //3.修改访问权限
            field.setAccessible(true);
            //4.创建类的实例
            Student student=(Student) c.newInstance();
            //5.设置属性值
            field.set(student,"白白");
            //6.获取属性值
            String name=(String) field.get(student);
            System.out.println(name);       //白白
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
    }



    /*
    * 通过反射调用私有方法
    *
    * */
    public static void reflectPrivateMethod(){
        try {
            //1.获取类对象

            Class<?> c = Class.forName("reflect.Student");
            //2.获取类中的某个方法,第一个参数代表类名,第二个参数代表方法的参数类型
            Method method=c.getDeclaredMethod("study",String.class);
            //3.对于私有的方法,修改权限
            method.setAccessible(true);
            Student student=(Student) c.newInstance();
            //根据参数进行匹配,调用对应的方法
            method.invoke(student,"白白");
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }
    }
    public static void main(String[] args) {
        //reflectNewInstance();
        reflectPrivateConstructor();
        //reflectPrivateField();
        //reflectPrivateMethod();
    }
}



几个重要的类及方法

  • Class类

在这里插入图片描述

  • Field类

在这里插入图片描述

  • Constructor类

在这里插入图片描述

  • Method类

在这里插入图片描述

反射的优缺点

优点:

对于任意一个类,都可以获取或调用类中的属性或方法;
反射机制增加了程序的扩展性,降低了耦合性;

缺点:

使用反射机制会降低程序的效率;
反射机制的代码更加复杂,维护成本高;

over!

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
### 回答1: Java反射机制是指在运行时动态地获取一个类的信息,并可以操作类的属性、方法和构造器等。Java反射机制可以使程序员在运行时动态地调用类的方法和属性,扩展类的功能,并可以实现注解、工厂模式以及框架开发等。 Java反射机制的原理如下:首先,Java编译器将Java源代码编译为字节码文件,字节码文件中包含着类的信息,这些信息包括类的名称、方法、属性和构造器等等。接着,Java虚拟机将字节码文件加载到内存中,然后通过类加载器将类加载到内存中形成一个类对象,这个类对象可以操作字节码文件中的信息。 使用Java反射机制的过程如下:首先获取类对象,通过类对象来获取类的构造器、属性、方法等信息,然后调用构造器来创建对象,通过属性获取和设置类的成员属性,通过方法调用类的方法等。 Java反射机制的优点是可以在运行时动态地得到类的信息,使得程序员在程序运行时能够对类进行更加灵活的操作,并可以使得程序更加通用化,同时也存在着一定的性能问题,因为Java反射机制需要Java虚拟机进行一定的额外处理,所以在程序运行时需要进行额外的时间和资源消耗。 总之,Java反射机制Java语言的一项重要特性,在Java开发中广泛应用,在代码编写、框架开发以及API开发中具有重要作用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值