参考到的三篇博客:
http://blog.csdn.net/ljphhj/article/details/12858767
http://blog.csdn.net/jiangwei0910410003/article/details/17733153
http://blog.csdn.net/gjanyanlig/article/details/6818655/
java类加载到内存中的过程
每个Java程序执行前都必须经过编译、加载、连接、和初始化这几个阶段,链接又分为三个步骤,如下图所示:
java程序运行在内存中,新建一个java对象的时候,JVM要将这个对象对应的字节码加载到内存中,JVM使用java类的方式如下:java源程序(.java文件)在经过java编译器之后就被转换成java字节代码(.class文件)。类加载器负责读取java字节代码,并转换成java.lang.class类的一个实例。每个这样的实例用来表示一个java类。通过此实例的newInstance()方法乐意创建出该类的一个对象。
加载(装载)
类的加载指将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个这个类的java.lang.class对象,用来封装在方法区类的对象。类的加载的最终产品是位于堆中的class对象。
基本上所有的类加载器都是 java.lang.ClassLoader类的一个实例。系统默认的三个主要的类加载器如下。
- 引导类加载器(bootstrap class loader):它用来加载 Java 的核心库,是用原生代码来实现的,并不继承自 java.lang.ClassLoader。
- 扩展类加载器(extensions class loader):它用来加载 Java 的扩展库。Java 虚拟机的实现会提供一个扩展库目录。该类加载器在此目录里面查找并加载 Java 类。
- 系统类加载器(system class loader):它根据 Java 应用的类路径(CLASSPATH)来加载 Java 类。一般来说,Java 应用的类都是由它来完成加载的。
类加载器也是Java类,因为Java类的类加载器本身也是要被类加载器加载的,显然必须有第一个类加载器不是Java类,这个正是BootStrap,使用C/C++代码写的,已经封装到JVM内核中了,而ExtClassLoader和AppClassLoader是Java类。
Java虚拟机中的所有类加载器采用具有父子关系的树形结构进行组织,在实例化每个类加载器对象的时候,需要为其指定一个父级类加载器对象或者默认采用系统类加载器为其父级类加载。
采用这种树形的类加载机制的好处就在于:能够很好的统一管理类加载,首先交给上级,如果上级有了,就加载,这样如果之前已经加载过的类,这时候在来加载它的时候只要拿过来用就可以了,无需二次加载了。
http://blog.csdn.net/jiangwei0910410003/article/details/17733153
链接
验证:确保被加载类的正确性;
首先如果由编译器生成的class文件,它肯定是符合JVM字节码格式的,但是万一有高手自己写一个class文件,让JVM加载并运行,用于恶意用途,就不妙了,因此这个class文件要先过验证这一关,不符合的话不会让它继续执行的,也是为了安全考虑吧。
准备:为类的静态变量分配内存,并将其初始化为默认值;
private static int a = 10,给a分配内存,因为变量a是static的,所以此时a等于int类型的默认初始值0,即a=0。
解析:把类中的符号引用转换为直接引用。
把a的真正的值10赋给a,此时a=10。
初始化
只有这6中情况才会导致类的类的初始化 :
- 创建类的实例,也就是new一个对象
- 访问某个类或接口的静态变量,或者对该静态变量赋值
- 调用类的静态方法
- 反射(Class.forName(“com.lyj.load”))
- 初始化一个类的子类(会首先初始化子类的父类)
- JVM启动时标明的启动类,即文件名和类名相同的那个类
类的初始化步骤:
- 如果这个类还没有被加载和链接,那先进行加载和链接
- 假如这个类存在直接父类,并且这个类还没有被初始化(注意:在一个类加载器中,类只能初始化一次),那就初始化直接的父类(不适用于接口)
- 加入类中存在初始化语句(如static变量和static块),那就依次执行这些初始化语句。
java反射机制
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
反射机制就是可以把一个类,类的成员(函数,属性),当成一个对象来操作,也就是说,类,类的成员,我们在运行阶段还可以动态地去操作他们。
用到反射机制的地方:
- 工厂模式:Factory类中用反射的话,添加了一个新的类之后,就不需要再修改工厂类Factory了;
- 数据库JDBC中通过Class.forName(Driver).来获得数据库连接驱动;
- 分析类文件:毕竟能得到类中的方法等等;
- 访问一些不能访问的变量或属性:破解别人代码。
反射机制要用Class.forName(“com.mysql.jdbc.Driver”)来动态加载制定的类.
Class.forName使用类加载机制,返回一个类,newInstance() 后才创建一个对象
生成一个实例的两种方法(第二种方法相当于将第一种方法分成了两步):
(1)A a = new A();
(2)
Class t = Class.forName("package.A");
Object object = t.NewInstance();
new 和 Class.forName()的区别:
1. newInstance( )是一个方法,而new是一个关键字;
2. Class下的newInstance()的使用有局限,因为它生成对象只能调用无参的构造函数,而使用 new关键字生成对象没有这个限制。
一个很好的反射机制的例子: http://blog.csdn.net/ljphhj/article/details/12858767