反射中,Class.forName 和 classloader 的区别
https://blog.csdn.net/qq_27093465/article/details/52262340
java中class.forName()和classLoader都可用来对类进行加载。
类加载过程为: 加载-->链接-->初始化;
加载:通过累的全限定名获取二进制字节流,将二进制字节流转换成方法区中的运行时数据结构,在内存中生成Java.lang.class对象;
链接:执行下面的校验、准备和解析步骤,其中解析步骤是可以选择的;
校验:检查导入类或接口的二进制数据的正确性;(文件格式验证,元数据验证,字节码验证,符号引用验证)
准备:给类的静态变量分配并初始化存储空间;
解析:将常量池中的符号引用转成直接引用;
初始化:激活类的静态变量的初始化Java代码和静态Java代码块,并初始化程序员设置的变量值。
1、Class.forName(className)方法,源码如下:
//Class.forName(String className) 这是1.8的源码 public static Class<?> forName(String className) throws ClassNotFoundException { Class<?> caller = Reflection.getCallerClass(); return forName0(className, true, ClassLoader.getClassLoader(caller), caller); } //注意第二个参数,是指Class被loading后是不是必须被初始化。 不初始化就是不执行static的代码即静态代码
第二个参数代表是否需要初始化,默认为true代表进行初始化。一旦初始化,就会触发目标对象的 static块代码执行,static参数也也会被再次初始化。2、ClassLoader.getSystemClassLoader().loadClass(className)方法,源码如下:
public Class<?> loadClass(String name) throws ClassNotFoundException { return loadClass(name, false); }
第2个 boolean参数,表示目标对象是否进行链接,false表示不进行链接,由上面介绍可以,不进行链接意味着不进行包括初始化等一些列步骤,那么静态块和静态对象就不会得到执行。
结论:
class.forName()前者除了将类的.class文件加载到jvm中之外,还会对类进行解释,执行类中的static块。
classLoader只干一件事情,就是将.class文件加载到jvm中,不会执行static中的内容,只有在newInstance才会去执行static块。
问题:数据库链接为什么使用Class.forName(className)
JDBC Driver源码如下, 因为使用Class.forName(classname) 才能在反射回去类的时候执行static块。
static { try { java.sql.DriverManager.registerDriver(new Driver()); } catch (SQLException E) { throw new RuntimeException("Can't register driver!"); } }