我是南城余!阿里云开发者平台专家博士证书获得者!
欢迎关注我的博客!一同成长!
一名从事运维开发的worker,记录分享学习。
专注于AI,运维开发,windows Linux 系统领域的分享!
本章节对应知识库
反射机制 · 语雀
Java给我们提供的一套API,使用这套API可以在运行时动态获取指定对象所属的类,创建运行时类的对象,调用指定的结构(属性、方法)等。
Reflection(反射)是被视为动态语言的关键,反射机制允许程序在运行期间借助于Reflection API取得任何类的内部信息,并能直接操作任意对象的内部属性及方法。
不使用反射,我们需要考虑封装性。比如出了Person类之后,就不能调用Person类中私有的结构
使用反射,我们可以调用运行时类中任意的构造器、属性、方法。包括了私有的属性、方法、构造器。
》从作为开发者角度,我们开发中主要是完成业务代码,对于相关的对象、方法的调用都是确定的。所以在开发中,我们使用非反射的方式多一些。
》因为反射体现了动态性(可以在运行时动态的获取对象所属的类,动态的调用相关的方法),所以我们在涉及框架时,会使用大量的反射。意味着,如果需要学习框架源码时,那么就需要学习反射。
框架 = 注解+反射+设计模式
封装性:体现的是是否建议我们调用内部api的问题。比如,private声明的结构,意味着不建议调用
反射: 体现的是我们能否调用的问题。因为类的完整结构都加载了内存中,所以我们就有能力进行调用
优点:
》提高了Java程序的灵活性和扩展性,降低了耦合性,提高了自适应能力
》允许程序创建个控制任何类的对象,无需提前硬编码目标类
缺点:
》反射的性能较低
反射机制主要应用在对灵活性和扩展性要求很高的系统框架上
》反射会模糊程序内部逻辑,可读性较差
反射,平时开发中,我们使用的并不多。主要是在框架的底层使用
针对于编写好的。java源文件进行编译(使用javac.exe)会生成一个或多个.class字节码文件。接着,我们使用java.exe命令对指定的.class文件进行解释运行。在这个解释运行的过程中,我们需要将.class字节码文件加载(使用类的加载器)到内存中(存在方法区)。加载到内存中的.class文件对应的结构即为Class的一个实例。
比如:加载到内存中的Person类或String类,都作为Class的一个一个的实例
Class clazz1 = Person.class;
Class clazz1 =String.class;
class可以看作是反射的源头
方式1:要求编译期间已知类型
前提:若已知具体的类,通过类的class属性获取,该方法最为安全可靠,程序性能最高
实例:
Class clazz = String.class;
方式2:获取对象的运行时类型
前提:已知某个类的实例,调用该实例的getClass()方法获取Class对象
实例:
Class clazz = "www.atguigu.com".getClass();
方式3:可以获取编译期间未知的类型
前提:已知一个类的全类名,且该类在类路径下,可通过Class类的静态方法forName()获取,可能抛出ClassNotFoundException
实例:
Class clazz = Class.forName("java.lang.String");
方式4:其他方式(不做要求)
前提:可以用系统类加载对象或自定义加载器对象加载指定路径下的类型
实例:
ClassLoader cl = this.getClass().getClassLoader();
Class clazz4 = cl.loadClass("类的全类名");
简言,所有的Java类型
》class:外部类,成员(成员内部类,静态内部类),局部内部类,匿名内部类
》interface:接口
》[]:数组
》enum:枚举
》annotation:注解@interface
》primitive type :基本数据类型
》void
过程1:类的装载(loading)
将类的class文件读入内存,并为之创建一个java.lang.Class对象。此过程由类加载器完成
过程2:链接(linking)
> 验证(Verify):确保加载的类信息符合JVM规范,例如:以cafebabe开头,没有安全方面的问题。
> 准备(Prepare):正式为类变量(static)分配内存并设置类变量默认初始值的阶段,这些内存都将在方法区中进行分配。
> 解析(Resolve):虚拟机常量池内的符号引用(常量名)替换为直接引用(地址)的过程。
过程3:初始化(initialization)
执行类构造器
类构造器
作用:负责类的加载,并对应于一个Class的实例。
分类(分为两种):
> BootstrapClassLoader:引导类加载器、启动类加载器
> 使用C/C++语言编写的,不能通过Java代码获取其实例
> 负责加载Java的核心库(JAVA_HOME/jre/lib/rt.jar或sun.boot.class.path路径下的内容)
> 继承于ClassLoader的类加载器
> ExtensionClassLoader:扩展类加载器
> 负责加载从java.ext.dirs系统属性所指定的目录中加载类库,或从JDK的安装目录的jre/lib/ext子目录 下加载类库
> SystemClassLoader/ApplicationClassLoader:系统类加载器、应用程序类加载器
> 我们自定义的类,默认使用的类的加载器。
> 用户自定义类的加载器
> 实现应用的隔离(同一个类在一个应用程序中可以加载多份);数据的加密。
以上的类的加载器是否存在继承关系? No!
/*
* 需求:通过ClassLoader加载指定的配置文件
* */
@Test
public void test3() throws IOException {
Properties pros = new Properties();
//通过类的加载器读取的文件的默认的路径为:当前module下的src下
InputStream is = ClassLoader.getSystemClassLoader().getResourceAsStream("info1.properties");
pros.load(is);
String name = pros.getProperty("name");
String pwd = pros.getProperty("password");
System.out.println(name + ":" +pwd);
}
如何实现
通过Class的实例调用newInstance()方法即可
且需要满足以下条件:
》要求运行时必须提供一个空参构造器
》要求提供的空参构造器的权限要足够
JavaBean中要求给当前类提供一个公共的的空参的构造器。
作用:
>场景1:子类对象在实例化时,子类的构造器的首行默认调用父类空参构造器
>场景2:在反射中,经常用来创建运行时类的对象。那么我们要求各个运行时类都提供一个空参构造器,便于我们编写创建运行时类对象的代码。
》获取运行时类的内部结构:所有属性、所有方法、所有构造器
》获取运行时类的内部结构:父类、接口、包、带泛型的父类、父类的泛型等
调用指定的属性步骤
步骤1. 通过Class实例调用getDeclareField(String fieldName),获取运行时类指定名的属性
步骤2. setAccessible(true),确保此属性是可以访问的
步骤3. 通过Field类的实例调用get(Object obj)(获取操作)
或set(Object obj,Object value)(设置的操作)进行操作
调用指定的方法步骤
步骤1. 通过Class实例调用getDeclareField(String methodName,Class ... args),获取运行时类指定的方法
步骤2. setAccessible(true),确保此属性是可以访问的
步骤3. 通过Method实例invoke(Object obj,Object .. objs),即为对Method对应方法的调用
invoke()返回值即为Method对应方法的返回值
特别的:如果Method对应的方法的返回值类型为void,则invoke()返回值为null
调用指定的构造器步骤
步骤1. 通过Class的实例调用getDeclaredConstructor(Class ... args),获取指定参数的构造器
步骤2. setAccessible(true):确保此构造器是可访问的
步骤3. 通过Constructor实例调用newInstance(Object ... objs),返回一个运行时类的实例
框架层面