Class的装载分了三个阶段,loading,linking和initializing,分别定义在The Java Language Specification的12.2,12.3和12.4。
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)。
分享到:
相关推荐
`Class.forName`和`ClassLoader`的区别在于使用场景和控制粒度。`Class.forName`更方便,它内部调用了`ClassLoader.loadClass`,并且自动处理了类的初始化。而`ClassLoader`提供了更底层的控制,可以用于更复杂的类...
3. **通过`ClassLoader.loadClass()`方法动态加载**:提供了更多的灵活性,可以根据不同的需求加载不同的类。 这三种方式各有特点,具体选择哪种方式取决于实际的应用场景。 #### 六、总结 Java的类加载机制是JVM...
4. 避免仅加载不初始化:值得注意的是,如果只是简单地使用`ClassLoader.loadClass()` 加载类,而没有实例化或执行静态初始化,那么驱动不会被注册到`DriverManager`。这样,即使类被加载到JVM中,也无法通过`...
at java.lang.ClassLoader.loadClass(Unknown Source) at sun.misc.Launcher$AppClassLoader.loadClass(Unknown Source) at java.lang.ClassLoader.loadClass(Unknown Source) at java.lang.Class.forName0...
at java.lang.ClassLoader.loadClass(Unknown Source) at java.lang.Class.forName0(Native Method) at java.lang.Class.forName(Unknown Source) at ...
at java.lang.ClassLoader.loadClass(Unknown Source) at java.lang.Class.forName0(Native Method) at java.lang.Class.forName(Unknown Source) at ...
`Class.forName()`可以初始化类的静态属性和方法,而`ClassLoader.loadClass()`则不会。例如: ```java // 显式加载 Class.forName("com.anbai.sec.classloader.TestHelloWorld"); // 自定义ClassLoader加载 ...
- 使用`ClassLoader.loadClass(String name)`方法显式加载类。 其中,`Class.forName()`方法有两种重载形式,第一个只接受类名作为参数,第二个允许控制是否初始化类以及指定类加载器。如果只使用一个参数的`Class...
这个问题通常发生在使用`Class.forName()`或`ClassLoader.getSystemClassLoader().loadClass()`尝试动态加载第三方JAR中的类时。 Java虚拟机(JVM)的类加载机制是导致此问题的关键。自JDK 1.2以来,JVM采用委托...
- **反射机制**通常涉及使用`Class.forName()`方法或`Constructor.newInstance()`方法来创建对象。 - **自定义类加载器**则是通过重写`loadClass`方法来自定义类的加载过程。 ### 总结 本实例展示了如何通过自定义...
- 通过`ClassLoader`的`loadClass`方法。 - 调用静态方法`Class.forName()`。 - 使用`Object`类的`getClass()`方法。 - 使用`.class`表达式直接获取类的`Class`对象。 2. 对于基本数据类型和`void`: - 使用`....
- **ClassLoader.loadClass()**:通过类加载器(如系统类加载器)的`loadClass()`方法,也能获取`Class`对象,例如`ClassLoader.getSystemClassLoader().loadClass("com.anbai.sec.classloader.TestHelloWorld")`。...
在自定义ClassLoader时,主要需要覆写两个关键方法:`findClass()` 和 `loadClass()`。`loadClass()` 方法通常用于委托父类加载器加载类,如果父类加载器无法加载,再由当前类加载器尝试加载。`findClass()` 方法则...
1. 继承ClassLoader:开发者可以通过继承`java.lang.ClassLoader`并重写`loadClass()`方法来自定义类加载逻辑。 2. 加载自定义位置的类:自定义ClassLoader可以加载非标准路径的类,如从网络、数据库或其他存储介质...
1. **递归文件系统扫描**:遍历指定目录下的所有文件,通过文件名判断是否为Class文件,然后使用`Class.forName()`或`ClassLoader.loadClass()`加载并检查。 2. **使用ClassLoader资源流**:通过`ClassLoader.get...
总结来说,Java中实现静默加载Class的方法主要有两种:一是使用`ClassLoader`的`loadClass(String name)`,二是调用`Class.forName(String name, boolean initialize, ClassLoader loader)`并传入`initialize = ...
当调用`Class.forName()`或`ClassLoader.loadClass()`时,JVM会使用类加载器找到对应的类。 反射API是实现动态类加载的关键工具。`java.lang.reflect`包提供了`Class`类、`Constructor`类、`Method`类和`Field`类等...
在编程中,我们有时会用到`Class.forName()`和`ClassLoader.loadClass()`这两个方法。`Class.forName()`不仅加载类,还会执行类的静态初始化块,而`ClassLoader.loadClass()`仅加载类,不执行静态初始化。`Class.for...
12. **Class.forName和ClassLoader.loadClass的区别**: - Class.forName会初始化类,而ClassLoader.loadClass只加载类但不初始化。 13. **HashMap与HashTable的区别**: - HashMap是AbstractMap的子类,...