初始java中的反射机制(二)
一、类的加载与ClassLoader的理解
当程序主动使用某个类时,如果该类还未被加载到内存中,则系统会通过如下三个步骤来对该类进行初始化。
①加载:将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区的运行时数据结构,然后生成一个代表这个类的java.lang.Class对象,作为方法区中类数据的访问入口(即引用地址)。所有需要访问和使用类数据只能通过这个Class对象。这个加载的过程需要类加载器参与。
②链接:将Java类的二进制代码合并到JVM的运行状态之中的过程。
验证:确保加载的类信息符合JVM规范,例如:以cafe开头,没有安全方面的问题
准备:正式为类变量(static)分配内存并 设置类变量默认初始值的阶段,这些内存
都将在方法区中进行分配。
解析:虚拟机常量池内的符号引用(常量名)替换为直接引用(地址)的过程。
③初始化:
执行类构造器()方法的过程。类构造器()方法是由编译期自动收集类中所有类变量的赋 值动作和静态代码块中的语句合并产生的。(类构造器是构造类信息的,不是构造该类对象的构器)。
当初始化一个类的时候,如果发现其父类还没有进行初始化,则需要先触发其父类的初始化。
虚拟机会保证一个类的()方法在多线程环境中被正确加锁和同步。
二、类加载器的作用及分类
类加载器的作用:
1、类加载的作用:将class文件字节码内容加载到内存中,并将这些静态数据转换成方
法区的运行时数据结构,然后在堆中生成一个代表这个类的java.lang.Class对象,作为
方法区中类数据的访问入口。
2、类缓存:标准的JavaSE类加载器可以按要求查找类,但一旦某个类被加载到类加载器
中,它将维持加载(缓存)一段时间。不过JVM垃圾回收机制可以回收这些Class对象。
引导类加载器:用C++编写的,是JVM自带的类加载器,负责Java平台核心库,用来装载核心类
库。该加载器无法直接获取
扩展类加载器:负责jre/lib/ext目录下的jar包或 –D java.ext.dirs 指定目录下的jar包装入工作库
系统类加载器:负责java –classpath 或 –Djava.class.path所指的目录下的类与jar包装入工 作 ,是最常用的加载器
public void test01() {
//对于自定义类,使用系统类加载器
ClassLoader loader1 = classLoderTest.class.getClassLoader();
System.out.println(loader1);//sun.misc.Launcher$AppClassLoader@18b4aac2
//调用系统类加载器的getParent(),可以获得扩展类加载器
ClassLoader loader2 = loader1.getParent();
System.out.println(loader2);//sun.misc.Launcher$ExtClassLoader@a09ee92
//调用扩展类加载器的getParent(),无法获得引导类加载器
//引导类加载器,主要加载java的核心类库,无法加载自定义类
ClassLoader loader3 = loader2.getParent();
System.out.println(loader3);//null
ClassLoader loader = String.class.getClassLoader();
System.out.println(loader);//null
}
关于类加载器的一个主要方法:getResourceAsStream(String str):获取类径下的指定文件的输入流此时配置文件的相对路径,是相对于src
@Test
public void test04() {
Properties properties = new Properties();
ClassLoader loader = classLoderTest.class.getClassLoader();
InputStream is = loader.getResourceAsStream("jdbc1.properties");
try {
properties.load(is);
} catch (IOException e) {
e.printStackTrace();
}
String user = properties.getProperty("user");
String password = properties.getProperty("password");
System.out.println("user=" + user + ",password=" + password);
try {
if (is != null) {
is.close();
}
} catch (IOException e) {
e.printStackTrace();
}
}