Class.forName() ClassLoader.loadClass () 区别小议
2015-12-22 16:04
176 查看
Class.forName(String className)。
Class t =Class.forName("java.lang.Thread")
调用forName("X") 将导致名为 X 的类被初始化。
[java] view
plaincopy
1.Class c1 = Class.forName (String className);
2.ClassLoadercl = new ClassLoader();
Class c1= cl.loadClass(String className, boolean resolve );
classLoader.loadClass(StringclassName , boolean resolve);需要手动指定类装载器的实例。
所以这两种类装载方式的区别之一是一个默认使用装载当前类实例的类装载器来装载指定类,而另一个则需要手动指定一个类装载器的实例。
上文提到Class.forName(className)方法内部调用了Class.forName(className, true, this.getClass().getClassLoader())方法。注意第二个参数,它指定Class被loading后是否要进行initializing操作。
ClassLoader.loadClass(className)方法内部调用ClassLoader.loadClass(name,false)方法,第二个参数指出Class在load之后,是否进行link操作。
区别就出来了。Class.forName(className)装载的class已经被实例化,而ClassLoader.loadClass(className)装载的class还没有被link,所以就更谈不上实例化了。一般情况下,这两个方法效果一样,都能装载Class。但如果程序需要Class被实例化,就必须用Class.forName(name)了。
打开com.mysql.jdbc.Driver的源代码看看,
// Register ourselves with the DriverManager //
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的初始化中被执行。所以这个地方就只能用Class.forName(className)。
官方文档
初始化参数指定的类,并且返回此类对应的Class 对象(不了解Class类的同志,应该首先查看下java反射相关知识)。例如,以下代码片段返回 java.lang.Thread 类对应的Class对象。Class t =Class.forName("java.lang.Thread")
调用forName("X") 将导致名为 X 的类被初始化。
类装载
类装载指通过解析类(或者接口)的字节码文件来构建代表这个类(或是这个接口)的实例的过程。有些像根据蓝图制作一个产品一样的概念。类装载有如下的两种方式:[java] view
plaincopy
1.Class c1 = Class.forName (String className);
2.ClassLoadercl = new ClassLoader();
Class c1= cl.loadClass(String className, boolean resolve );
两种装载方法的区别:
不同的类装载器
Class.forName(String className)使用装载当前类的类装载器来装载指定类。因为class.forName(String className)方法内部调用了Class.forName(className,true, this.getClass().getClassLoader())方法,如你所见,第三个参数就是指定类装载器,显而易见,它指定的是装载当前类的类装载器的实例,也就是this.getClass().getClassLoader();classLoader.loadClass(StringclassName , boolean resolve);需要手动指定类装载器的实例。
所以这两种类装载方式的区别之一是一个默认使用装载当前类实例的类装载器来装载指定类,而另一个则需要手动指定一个类装载器的实例。
是否实例化类
Class的装载过程(也就是从字节码文件到生成类的实例这个过程)分为三个阶段,loading(装载),linking(连接)和initializing(实例化)。上文提到Class.forName(className)方法内部调用了Class.forName(className, true, this.getClass().getClassLoader())方法。注意第二个参数,它指定Class被loading后是否要进行initializing操作。
ClassLoader.loadClass(className)方法内部调用ClassLoader.loadClass(name,false)方法,第二个参数指出Class在load之后,是否进行link操作。
区别就出来了。Class.forName(className)装载的class已经被实例化,而ClassLoader.loadClass(className)装载的class还没有被link,所以就更谈不上实例化了。一般情况下,这两个方法效果一样,都能装载Class。但如果程序需要Class被实例化,就必须用Class.forName(name)了。
在jdbc链接数据库中的应用
在JDBC中加载驱动类时(关于加载jdbc驱动请参看另外一篇文章,<jdbc注册驱动的三种方式>http://blog.csdn.net/yanwushu/article/details/7574708),Class.forName("com.mysql.jdbc.Driver"),如果换成getClass().getClassLoader().loadClass("com.mysql.jdbc.Driver"),就不行,因为它只是向jvm装载了Driver的类信息,但是并没有实例化,也就不能执行相应的操作,因为Driver是需要被初始化才能使用的。打开com.mysql.jdbc.Driver的源代码看看,
// Register ourselves with the DriverManager //
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的初始化中被执行。所以这个地方就只能用Class.forName(className)。
相关文章推荐
- 关于DRM文件
- 第七章 注解式控制器的数据验证、类型转换及格式化(1)
- Python学习笔记——函数式编程
- delete 和truncate 区别
- 在 Cent OS 6.5 中安装桌面环境
- 在广播中启动activity或者dialog
- 31.Evaluate the following SQL commands:
- GeoHash核心原理解析及java代码实现(转)
- 面对对象语言的三大特征
- 数据库4-修改数据表
- EasyUI序列化提交学习总结
- Web Logic 中间件(一)
- 常用的抽象类与实现类
- 嵌入式Linux串口应用编程基础知识
- 第六章 注解式控制器详解(5)
- C语言
- 2015-12-13复习之CSS3背景background的几个属性
- dstat
- css实现各种图形
- C语言总结