Class的装载分了三个阶段,loading,linking和initializing。java
Class.forName(xxx.xx.xx)加载对应类的字节码,返回的是一个类 ,静态代码是和class绑定的class装载成功就表示执行了你的静态代码了,也就是说JVM会执行该类的静态代码段并且之后不会再走这段静态代码了。调用此方法等效于Class.forName(className, true, this.getClass().getClassLoader())注意第二个参数,是指Class是否被link。Class.forName(className)装载的class已经被link,而ClassLoader.loadClass(className)装载的class尚未被link。ClassLoader.loadClass方法只是将class加载到JVM中,并不会执行其静态初始化代码。mysql
Class.forName( )静态方法的目的是为了动态加载类。在加载完成后,通常还要调用Class下的newInstance( )静态方法来实例化对象以便操做。sql
A a = (A)Class.forName("package.A").newInstance()和 A a = new A()是同样的效果。数据库
从JVM的角度看,咱们使用关键字new建立一个类的时候,这个类能够没有被加载。可是使用newInstance()方法的时候,就必须保证:
一、这个类已经加载;
二、这个类已经链接了。
ide
而完成上面两个步骤的正是Class的静态方法forName()所完成的,这个静态方法调用了启动类加载器,即加载 java API的那个加载器。this
那么为何会有两种建立对象方式?这主要考虑到软件的可伸缩、可扩展和可重用等软件设计思想。 spa
Java中工厂模式常常使用newInstance()方法来建立对象,所以从为何要使用工厂模式上能够找到具体答案。 设计
例如: xml
class c = Class.forName(“Example”); factory = (ExampleInterface)c.newInstance();
其中ExampleInterface是Example的接口,能够写成以下形式: 对象
String className = "Example"; class c = Class.forName(className); factory = (ExampleInterface)c.newInstance();
进一步能够写成以下形式:
String className = readfromXMlConfig;//从xml 配置文件中得到字符串 class c = Class.forName(className); factory = (ExampleInterface)c.newInstance();
上面代码已经不存在Example的类名称,它的优势是,不管Example类怎么变化,上述代码不变,甚至能够更换Example的兄弟类Example2 , Example3 , Example4……,只要他们继承ExampleInterface就能够。 咱们能够在调用class的静态加载方法forName时得到更好的灵活性,提供给了一种降耦的手段。
有数据库开发经验朋友会发现,为何在咱们加载数据库驱动包的时候有的却没有调用newInstance( )方法呢?即有的jdbc链接数据库的写法里是Class.forName(xxx.xx.xx);而有一些:Class.forName(xxx.xx.xx).newInstance(),为何会有这两种写法呢? 刚才提到,Class.forName("");的做用是要求JVM查找并加载指定的类,若是在类中有静态初始化器的话,JVM必然会执行该类的静态代码段。而在JDBC规范中明确要求这个Driver类必须向DriverManager注册本身,即任何一个JDBC Driver的Driver类的代码都必须相似以下:
public class MyJDBCDriver implements Driver { static { DriverManager.registerDriver(new MyJDBCDriver()); } }
}
打开com.mysql.jdbc.Driver的源代码看看,
static { try { java.sql.DriverManager.registerDriver(new Driver()); } catch (SQLException E) { throw new RuntimeException("Can't register driver!"); } }
原来,Driver在static块中会注册本身到java.sql.DriverManager。而static块就是在Class的初始化中被执行。既然在静态初始化器的中已经进行了注册,因此咱们在
使用JDBC时只须要Class.forName(XXX.XXX);就能够了。
newInstance: 弱类型。低效率。只能调用无参构造。 new: 强类型。相对高效。能调用任何public构造。