java反射机制快速入门及常见方法全归纳。(代码片段)

小黎的培培笔录 小黎的培培笔录     2022-10-23     646

关键词:

目录

一、反射机制

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框架有很大帮助,比如Spring框架的核心就是使用Java反射实现的,同时对做一些Java底层操作也会有帮助,比如字节码操作、自定义注解等。什么是反射Java反射说的是在运行状态中,对于任何一个类,... 查看详情

入门级java反射机制

1.什么是反射?Java中的反射机制是Java语言的一个很重要的特性,是Java“动态性”的重要体现。Java反射机制让我们在程序运行状态中,对于任意一个类,都能知道这个类的所有属性和方法;对于任意一个对象,都能调用他... 查看详情

反射机制入门(代码片段)

day34课堂笔记2、反射机制(比较简单,因为只要会查帮助文档,就可以了。)2.1、反射机制有什么用?通过java语言中的反射机制可以操作字节码文件。优点类似于黑客。(可以读和修改字节码文件。)... 查看详情

java的反射机制,看完这篇轻松应对高级框架(超详细总结)(代码片段)

导读:很多优秀的高级框架都是通过反射完成的,反射的重要性,由此可见一斑。反射机制可以使得程序更加灵活,只有学习好反射的基础语法,这样才能自己写出优秀的框架。好了一起打卡学习吧,别忘记了素质三连哦!往期... 查看详情

java反射机制(代码片段)

一、什么是反射?在运行状态中,对于任意一个类,都能够获取到这个类的所有属性和方法,对于任意一个对象,都能够调用它的任意一个方法和属性(包括私有),这种动态获取的信息,及调用对象的方法的功能就称为Java的... 查看详情

最常见的java面试题及答案汇总(代码片段)

反射57.什么是反射?反射主要是指程序可以访问、检测和修改它本身状态或行为的一种能力Java反射:在Java运行时环境中,对于任意一个类,能否知道这个类有哪些属性和方法?对于任意一个对象,能否调用它的任意一个方法Java... 查看详情

最常见的java面试题及答案汇总(代码片段)

反射57.什么是反射?反射主要是指程序可以访问、检测和修改它本身状态或行为的一种能力Java反射:在Java运行时环境中,对于任意一个类,能否知道这个类有哪些属性和方法?对于任意一个对象,能否调用它的任意一个方法Java... 查看详情

java常见集合的默认大小及扩容机制

...题,不仅要知道各集合的区别用法,还要知道集合的扩容机制,今天我们就来谈下ArrayList和HashMap的默认大小以及扩容机制。在Java7中,查看源码可以知道:ArrayList的默认大小是10个元素,HashMap的默认大小是16个元素(必须是2的幂... 查看详情

java中的反射

一、什么是反射  简单来说,java反射机制其实就是I/O流的一种封装版,用来快速读取硬盘上的class文件。class文件相当于一个身份证,JVM在操作某一个对象时,需要根据身份证获得对象拥有的属性和拥有的功能及方法。这种动... 查看详情

反射机制入门(代码片段)

day34课堂笔记2、反射机制(比较简单,因为只要会查帮助文档,就可以了。)2.1、反射机制有什么用?通过java语言中的反射机制可以操作字节码文件。优点类似于黑客。(可以读和修改字节码文件。)... 查看详情

java之反射—用法及原理

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

反射机制(reflection)(代码片段)

一、Java反射机制概述1.准动态语言动态语言可以在运行时,根据某些条件改变自身结构Object-C、C#、JavaScript、PHP、Python静态语言运行时结构不可改变Java、C、C++利用反射机制可以使Java获得类似动态语言的特性2.反射反射机制在执... 查看详情

java反射机制及ioc原理

 一. 反射机制概念  主要是指程序可以访问,检测和修改它本身状态或行为的一种能力,并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义。在java中,只要给定类的名字,那么就可以通... 查看详情

java反射机制及ioc原理

一. 反射机制概念  主要是指程序可以访问,检测和修改它本身状态或行为的一种能力,并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义。在java中,只要给定类的名字,那么就可以通过反... 查看详情

反射机制(代码片段)

...动态获取的信息及动态调用对象的方法的功能称为java的反射机制为什么要用反射机制,直接创建对象不就可以了?这就涉及到了动态与静态的概念  静态编译:在编译时确定类型,定对象  动态编辑:在运行时确定类型,绑定对象... 查看详情

java快速入门-03-小知识汇总篇(全)

Java快速入门-03-小知识汇总篇(全)前两篇介绍了JAVA入门的一系小知识,本篇介绍一些比较偏的,说不定什么时候会用到,有用记得Mark一下快键键常用快捷键(熟记)快捷键快捷键作用Ctrl+Shfit+O引入及管理import语句Ctrl+Shfit+F4/Ctr... 查看详情

java反射及ioc原理内省机制

JAVA反射及IoC原理、JAVA内省1.反射反射是框架设计的灵魂,使用前提:必须先得到代表的字节码的Class,Class类用于表示.class文件(字节码文件)。1.1反射概述主要指程序可以访问,检测和修改它本身状态或行为的一种能力,并能根... 查看详情

java自用高级编程-8.反射机制(代码片段)

...段的最后一章~~~~~~~~~JavaSE终于要结束了。。。。。Java的反射机制一、反射的概述1.本章的主要内容(红色需要掌握)2.关于反射的理解Reflection(反射)是被视为动态语言的关键,反射机制允许程序在执行期间借助于... 查看详情