Class.forName和ClassLoader.loadClass的区别

来源:互联网 发布:没有域名需要备案吗 编辑:程序博客网 时间:2024/05/21 18:50

Class.forName(String className)使用装载当前类的类装载器来装载指定类。在Class.forName(String className)方法内部调用了Class.forName(className,true, this.getClass().getClassLoader())方法,如你所见,第三个参数就是指定类装载器,显而易见,它调用的是装载当前类的类装载器实例(也就是this.getClass().getClassLoader()); ClassLoader.loadClass(String className , boolean resolve) 需要手动new一个类装载器实例,再去loadClass。 所以这两种类装载方式的区别之一就是一个默认通过当前类实例的类装载器来装载指定类,而另一个则需要手动new一个类装载器实例再去装载指定类。

分析: Class的装载分了三个阶段,loading,linking和initializing。 Class.forName(className)实际上是调用Class.forName(className, true, this.getClass().getClassLoader())。注意第二个参数,是指Class被loading后是不是必须被初始化。 ClassLoader.loadClass(className)实际上调用的是ClassLoader.loadClass(name, false),第二个参数指出Class是否被link。 区别就出来了。Class.forName(className)装载的class已经被初始化,而ClassLoader.loadClass(className)装载的class还没有被link,就更加谈不上初始化了。 一般情况下,这两个方法效果一样,都能装载Class。但如果程序依赖于Class是否被初始化,就必须用Class.forName(name)了。 例如,在JDBC编程中,常看到这样的用法,Class.forName(“com.mysql.jdbc.Driver”),如果换成了getClass().getClassLoader().loadClass(“com.mysql.jdbc.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)。

然而,类装载其可以满足一些forName()无法满足的需求。如果需要一些特定的装载类型的方法,比如从网络上下载,从数据库中取出,从加密文件中哦功能提取,甚至动态创建它们,这时就需要一个类装载其。创建用户自定义的类装载器,其中一个重要的原因就是能够以定制方式把类型的全限定名转换成一个java class文件格式(它定义了命名的类型)的字节数组。使用类装载其而非forName()的其他理由和安全相关。每一个类装载器拥有一个独立的命名空间,这就为不同的命名空间中装载的类型提供了一层安全防护。可以编写一个java程序,类型无法看见不在同一命名空间装载的其他类型。类装载器负责把装载的代码放到保护域中,也就是说,如果安全上需要包含一种定制方式把类型装载到保护域着哦功能,就需要使用类装载其而非forName()。

阅读全文
0 0
原创粉丝点击