Java 反射机制快速入门及常见方法全归纳。

Posted 小黎的培培笔录

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java 反射机制快速入门及常见方法全归纳。相关的知识,希望对你有一定的参考价值。

目录

一、反射机制

1、基本介绍

2、原理示意图

3、反射基本代码实现

4、反射性能

 二、Class 类

1、基本介绍

2、获取 Class类对象的方式

3、有 Class对象的类

三、类加载

1、基本介绍

2、连接阶段

四、常见方法取类的结构信息

1、常用类的方法

2、通过反射创建对象


一、反射机制

1、基本介绍

        基本概念:在计算机科学领域,反射是指一类应用,它们能够自描述和自控制。也就是说,这类应用通过采用某种机制来实现对自己行为的描述(self-representation)和监测(examination),并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义。

        基于Java:Java中,反射是一种强大的工具。它使您能够创建灵活的代码,这些代码可以在运行时装配,无需在组件之间进行源代表链接。反射允许我们在编写与执行时,使我们的程序代码能够接入装载到JVM中的类的内部信息,而不是源代码中选定的类协作的代码。这使反射成为构建灵活的应用的主要工具。但需注意的是:如果使用不当,反射的成本很高。

        注意:

            1、反射机制允许程序在执行期借助于(Reflection  API)取得任何类的内部信息(比如成员变量,构造器,成员方法等等), 并能操作对象的属性及方法。 反射在设计模式和框架底层都会用到。
            2、加载完类之后,在堆中就产生了一个Class类型的对象(一个类只有一个Class对象),这个对象包含了类的完整结构信息。 通过这个对象得到类的结构。 这个Class对象就像一面镜子,透过这个镜子看到类的结构,所以,形象的称之为:反射。

2、原理示意图

 

 ▶ 发射机制的应用

        1、在运行时判断任意一个对象所属的类

        2、在运行时构造任意一个类的对象
        3、在运行时得到任意一个类所具有的成员变量和方法

        4、在运行时调用任意一个对象的成员变量和方法

        5、生成动态代理

 ▶ 反射常用类

        1、java.lang.Class : 代表一个类, Class对象表示某个类加载后在堆中的对象
        2、java.lang.reflect. Method : 代表类的方法,Method对象表示某个类的方法
        3、java.lang.refiect.Field : 代表类的成员变量,Field对象表示某个类的成员变量

        4、java.lang.reflect.Constructor : 代表类的构造方法,Constructor对象表示构造器

3、反射基本代码实现

//(1) 加载类, 返回 Class 类型的对象 cls

Class cls = Class.forName(classfullpath);

//(2) 通过 cls 得到你加载的类 Cat 的对象实例

Object o = cls.newInstance();

//(3) 通过 cls 得到你加载的类 Cat 的 methodName"hi" 的方法对象

// 即:在反射中,可以把方法视为对象(万物皆对象)

Method method1 = cls.getMethod(methodName);

//(4) 通过 method1 调用方法: 即通过方法对象来实现调用方法

method1.invoke(o); //传统方法:对象.方法() , 反射机制:方法.invoke(对象)
//java.lang.reflect.Field: 代表类的成员变量, Field 对象表示某个类的成员变量

//注意:getField 不能得到私有的属性

Field nameField = cls.getField("age");//得到 name 字段

//传统写法:对象.成员变量 , 反射:成员变量对象.get(对象)
System.out.println(nameField.get(o)); 

//java.lang.reflect.Constructor: 代表类的构造方法, Constructor 对象表示构造器
//()中可以指定构造器参数类型, 此处返回无参构造器
Constructor constructor = cls.getConstructor(); 

System.out.println(constructor); //Cat()

//String.class 就是 String 类的 Class 对象,有参构造器
Constructor constructor2 = cls.getConstructor(String.class); 

System.out.println(constructor2); //Cat(String name)

4、反射性能

▶ 反射的优缺点

        1、优点:可以动态的创建和使用对象(也是框架底层核心),使用灵活,没有反射机制, 框架技术就失去底层支撑。
        2、缺点:使用反射基本是解释执行,对执行速度有影响。

▶ 传统方式调用方法

public static void m1() 

    Cat cat = new Cat();

    long start = System.currentTimeMillis();

    for (int i = 0; i < 90; i++) 
        cat.hi();
    

    long end = System.currentTimeMillis();

    System.out.println("m1() 耗时=" + (end - start));

▶ 反射方式调用方法

public static void m2() throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException 

    Class cls = Class.forName("com.Cat");

    Object o = cls.newInstance();

    Method hi = cls.getMethod("hi");

    long start = System.currentTimeMillis();

    for (int i = 0; i < 900000000; i++) 
        hi.invoke(o);//反射调用方法
    

    long end = System.currentTimeMillis();

    System.out.println("m2() 耗时=" + (end - start))

▶ 反射优化

        ▷ Method 和 Field,Constructor 对象都有 setAccessible()方法
        ▷ setAccessible 作用是启动和禁用访问安全检查的开关
        ▷ 参数值为true表示反射的对象在使用时取消访问检查,提高反射的效率。参数值为false则表示反射的对象执行访问检查

public static void m3() throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException 

    Class cls = Class.forName("com.hspedu.Cat");

    Object o = cls.newInstance();

    Method hi = cls.getMethod("hi");

    hi.setAccessible(true);//在反射调用方法时,取消访问检查

    long start = System.currentTimeMillis();

    for (int i = 0; i < 900000000; i++) 
        hi.invoke(o);//反射调用方法
    

    long end = System.currentTimeMillis();

    System.out.println("m3() 耗时=" + (end - start));


 二、Class 类

1、基本介绍

        ▶ Class也是类,因此也继承Object类
        ▶ Class类对象不是new出来的, 而是系统创建的
        ▶ 对于某个类的Class类对象, 在内存中只有一份。 因为类只加载一次
        ▶ 每个类的实例都会记得自己是由哪个Class实例所生成
        ▶ 通过Class对象可以完整地得到一个类的完整结构,通过一系列API
        ▶ Class对象是存放在堆的
        ▶ 类的字节码二进制数据, 是放在方法区的, 有的地方称为类的元数据(包括方法代码,变量名,方法名,访问权限等等) 

▶ 常见方法 

▶ 代码示例

String classAllPath = "com.Car";
//获取到 Car 类 对应的 Class 对象 。 <?> 表示不确定的 Java 类型

Class<?> cls = Class.forName(classAllPath);

//输出 cls
System.out.println(cls); //显示 cls 对象, 是哪个类的 Class 对象 com.Car

System.out.println(cls.getClass());//输出 cls 运行类型 java.lang.Class
//得到包名
System.out.println(  cls.getPackage().getName()  );//包名
//得到全类名
System.out.println(  cls.getName()  );
//5. 通过 cls 创建对象实例
Car car = (Car) cls.newInstance();
//6. 通过反射获取属性 brand
Field brand = cls.getField("brand");

System.out.println(  brand.get(car)  );//输出方式
//7. 通过反射给属性赋值
brand.set(car, "奔驰");

System.out.println(  brand.get(car)  );
//8 可以得到所有的属性(字段)
Field[] fields = cls.getFields();

for (Field f : fields) 

    System.out.println(f.getName());//名称

2、获取 Class类对象的方式

▶(1)第一种方式

  ▷ 前提 : 已知一个类的全类名,且该类在类路径下, 可通过Class类的静态方法forName()获取,可能抛出ClassNotFoundException。

  ▷ 实例 :

Class cls1 = Class.forName("java.lang.Cat")

   ▷ 应用场景 : 多用于配置文件,读取类全路径,加载类。

▶(2)第二种方式

  ▷ 前提 : 若已知具体的类,通过类的class 获取,该方式 最为安全可靠,程序性能
  ▷ 实例:

Class cls2 = Cat.class

   ▷ 应用场景 : 多用于参数传递,比如通过反射得到对应构造器对象

▶(3)第三种方式

  ▷前提 : 已知某个类的实例, 调用该实例的getClass0方法获取Class对象,

  ▷实例:

Class class = 对象.getClass(); //运行类型

  ▷应用场景 : 通过创建好的对象,获取Class对象.其他方式
►(4)其他方式

ClassLoader cl = 对象.getClass().getClassLoader();

Class clazz4 = cl.loadClass(“类的全类名”);

▶ (5)基本数据(int, charboolean, float,double, byte,long,short) 按如下方式得到Class类对象

Class cls = 基本数据类型.class

▶ (6)基本数据类型对应的包装类,可以通过.TYPE 得到Class类对象

Class cls = 包装类.TYPE

3、有 Class对象的类

        ▷ 外部类,成员内部类,静态内部类,局部内部类, 匿名内部类

        ▷ interface : 接口
        ▷ 数组
        ▷ enum : 枚举
        ▷ annotation : 注解基本数据类型
        ▷ void

Class<String> cls1 = String.class;//外部类

Class<Serializable> cls2 = Serializable.class;//接口

Class<Integer[]> cls3 = Integer[].class;//数组

Class<float[][]> cls4 = float[][].class;//二维数组

Class<Deprecated> cls5 = Deprecated.class;//注解

Class<Thread.State> cls6 = Thread.State.class;//枚举

Class<Long> cls7 = long.class;//基本数据类型

Class<Void> cls8 = void.class;//void 数据类型

Class<Class> cls9 = Class.class;


三、类加载

1、基本介绍

▶ 基本概念

        反射机制是java实现动态语言的关键,也就是通过反射实现类动态加载。①静态加载 :  编译时加载相关的类, 如果没有则报错, 依赖性太强。②动态加载 : 运行时加载需要的类,如果运行时不用该类, 即使存在该类,则也不会报错,降低了依赖性。
▶ 什么时候加载类

        ▷ 当创建对象时(new)。  //静态加载

        ▷ 当子类被加载时,父类也加载。 //静态加载

        ▷ 调用类中的静态成员时。 //静态加载

        ▷ 通过反射。 //动态加载,Class.forName("com.Cat");
 

▶ 类加载过程图

 ▶ 类加载各阶段图

▶ 加载阶段 

 2、连接阶段

▶验证阶段

   ▷目的是为了确保Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。
   ▷验证包括 :  文件格式验证(是否以魔数 oxcafebabe开头)、元数据验证、字节码验证和符号引用验证
   ▷可以考虑使用-Xverify : none 参数来关闭大部分的类验证措施, 缩短虚拟机类加载的时间。

▶准备阶段

   ▷ JVM 会在该阶段对静态变量,分配内存井默认初始化(对应数据类型的默认初始值,如 0、OL、 null, false 等),这些变量所使用的内存都将在方法区中进行分配。

//1. n1 是实例属性, 不是静态变量,因此在准备阶段,是不会分配内存

public int n1 = 10;
//2. n2 是静态变量,分配内存 n2 是默认初始化 0 ,而不是 20

public static int n2 = 20;
//3. n3 是 static final 是常量, 他和静态变量不一样, 因为一旦赋值就不变 n3 = 30

public static final int n3 = 30;

▶ 解析阶段

        虚拟机将常量池里面的符号引用替换为直接引用的的过程。

▶ 初始化

      ▷ 到初始化阶段, 才真正开始执行类中定义的Java程序代码,此阶段是执行<clinit>() 方法的过程。
      ▷ <clinit>()方法是由编译器按语句在源文件中出现的顺序,依次自动收集类中的所有静态变量的赋值动作和静态代码块中的语句,并进行合并。 
      ▷ 虚拟机会保证一个类的<clinit>()方法在多线程环境中被正确地加锁、同步,如果多个线程同时去初始化一个类,那么只会有一个线程去执行这个类的<clinit>()方法,其他线程都需要阻塞等待,直到活动线程执行<clinit>() 方法完毕。


四、常见方法取类的结构信息

1、常用类的方法

▶ java.lang.Class 类

方法名功能
getName获取全类名
getSimpleName获取简单类名
getFields获取所有public修饰的属性,包含本类以及父类的
getDeclaredFields获取本类中所有属性
getMethods获取所有public修饰的方法,包含本类以及父类的
getDeclaredMethods获取本类中所有方法
getConstructors获取本类所有public修饰的构造器
getDeclaredConstructors获取本类中所有构造器
getPackage以Package形式返回包信息
getSuperClass以Class形式返回父类信息
getinterfaces以Class[ ]形式返回接口信息
getAnnotations以Annotation[ ] 形式返回注解信息

▶ java.lang.reflect. Method 类

方法名功能
getModifiers以int形式返回修饰符。[说明:默认修饰符是0 ,public 是1,private是2 ,protected 是4 ,static是8, final 是 16]
getReturnType以Class形式获取 返回类型
getName返回方法名
getParameterTypes以Class返回参数类型数组

▶ java.lang.refiect.Field 类

方法名功能
getModifiers以int形式返回修饰符。[说明:默认修饰符是0 ,public 是1,private是2 ,protected 是4 ,static是8, final 是 16]
getType以Class形式返回类型
getName返回属性名

▶ java.lang.reflect.Constructor 类

方法名功能
getModifiers以int形式返回修饰符
getName返回构造器名(全类名)
getParameterTypes以Class[ ]返回参数类型数组

2、通过反射创建对象

▶ 方式一 : 调用类中的public修饰的无参构造器
▶ 方式二 : 调用类中的指定构造器
▶ Class类相关方法:
     ▷ newlnstance : 调用类中的无参构造器, 获取对应类的对象

     ▷ getConstructor(Class..clazz) : 根据参数列表,获取对应的public构造器对象              

     ▷ getDecalaredConstructor(Class..clazz) : 根据参数列表,获取对应的所有构造器对象
▶ Constructor类相关方法:

     ▷ setAccessible : 暴破
     ▷ newlnstance(Object..obj) : 调用构造器

▶ 代码实现

//1. 先获取到 User 类的 Class 对象 
Class<?> userClass = Class.forName("com.reflection.User"); 
//2. 通过 public 的无参构造器创建实例 
Object o = userClass.newInstance(); 
//3. 通过 public 的有参构造器创建实例,先得到对应构造器 
Constructor<?> constructor = userClass.getConstructor(String.class); 

//创建实例,并传入实参 
Object h = constructor.newInstance("hi"); 
//4. 通过非 public 的有参构造器创建实例,得到 private 的构造器对象 
Constructor<?> constructor1 = userClass.getDeclaredConstructor(int.class, String.class);
 
//创建实例,暴破【暴力破解】 , 使用反射可以访问 private 构造器/方法/属性
constructor1.setAccessible(true); 

Object user2 = constructor1.newInstance(传入对应参数...);

以上是关于Java 反射机制快速入门及常见方法全归纳。的主要内容,如果未能解决你的问题,请参考以下文章

最常见的Java面试题及答案汇总

最常见的Java面试题及答案汇总

入门级 JAVA反射机制

java的反射机制入门

Java常见问题

java中的反射