一、概念:JAVA反射机制是在运行状态中,获取任意一个类的结构 , 创建对象 , 得到方法,执行方法 , 属性 !;
这种在运行状态动态获取信息以及动态调用对象方法的功能被称为java语言的反射机制。
二、类加载器:
Java类加载器(Java Classloader)是Java运行时环境(Java Runtime Environment)的一部分,
负责动态加载Java类到Java虚拟机的内存空间中。
java默认有三种类加载器,BootstrapClassLoader、ExtensionClassLoader、App
ClassLoader。
BootstrapClassLoader(引导启动类加载器):
嵌在JVM内核中的加载器,该加载器是用C++语言写的,主要负载加载JAVA_HOME/lib下的类库,引导启动类加载器无法被应用程序直接使用。
ExtensionClassLoader(扩展类加载器):
ExtensionClassLoader是用JAVA编写,且它的父类加载器是Bootstrap。
是由sun.misc.Launcher$ExtClassLoader实现的,主要加载JAVA_HOME/lib/ext目录中的类
库。
它的父加载器是BootstrapClassLoader
App ClassLoader(应用类加载器):
App ClassLoader是应用程序类加载器,负责加载应用程序classpath目录下的所有jar和class文
件。它的父加载器为Ext ClassLoader
此处的子父类不像我们程序中的继承关系,而是一种委派关系。
面试题:多个类加载器之间是如何避免类加载重复的?
原因:双亲委派模型:
如果一个类加载器收到了一个类加载请求,它不会自己去尝试加载这个类,而是把这个请求转交给父类加载器去完成。每一个层次的类加载器都是如此。因此所有的类加载请求都应该传递到最顶层的启动类加载器中,只有到父类加载器反馈自己无法完成这个加载请求(在它的搜索范围没有找到这个类)时,子类加载器才会尝试自己去加载。委派的好处就是避免有些类被重复加载。
三、用类加载器加载配置文件
先创建一个文件:注意:直接创建在src下,不要创建在包里。
我们来用来加载器来加载这个文件:
public static void main(String[] args) throws IOException {
//1.通过类名找到class(字节码),再通过getClassLoader()获取类加载器(此处是App ClassLoader)
//拿到类加载器后,通过getResourceAsStream()获取在src下的文件的输入流
InputStream is = Demo4.class.getClassLoader().getResourceAsStream("config.txt");
//将字节流转换为便于读取的字符流
BufferedReader br = new BufferedReader(new InputStreamReader(is));
String text = br.readLine();
System.out.println(text);
br.close();
}
结果:
注意:在后续的框架中如果在项目中创建一个Sorce文件夹,若此文件夹被设置为Resources Root(资源根目录)。那么上面的getResourceAsStream("config.txt")获取的就不是src里的文件了,而是Sorce下的文件。