Class.forName和ClassLoader有啥区别?

  • Post author:
  • Post category:其他


Java中的Class.forName()和ClassLoader 都可以对类进行加载,在学习mysql期间可能会经常用到Class.forName(),ClassLoader负责将Class的字节码形式转换成内存形式的 Class 对象。字节码可以来自于磁盘文件,也就是XXXX.class,也可以来自jar包里的XXXX.class,也可以来自远程服务器提供的字节流,字节码的本质就是一个字节数组[]byte。

实际上Class.forName()方法也是调用的 ClassLoader 来实现的。

forName方法最后调用 forName0(本地方法),第二个参数被默认设置为了 true,这个参数代表是否对加载的类进行初始化,设置为 true 时会类进行初始化,代表会执行类中的静态代码块,以及对静态变量的赋值等操作。

 @CallerSensitive
 public static Class<?> forName(String className)
             throws ClassNotFoundException {
     Class<?> caller = Reflection.getCallerClass();
     return forName0(className, true, ClassLoader.getClassLoader(caller), caller);
 }

他还有个重载方法,可以手动选择在加载类时是否要对其类进行初始化。

  public static Class<?> forName(String name, boolean initialize,
                                 ClassLoader loader){}

编写一个类测试一下。

public class Test {
    static {
        System.out.println("静态代码快");
    }
    public static final String title=getTitle();

    private static String getTitle(){
        System.out.println("执行getTitle()");
        return "TITLE";
    }
}

测试方法:

public class Main {
    public static void main(String[] args) throws ClassNotFoundException {
        Class.forName("com.hxl.Test");
    }
}

运行结果:

静态代码快
执行getTitle()

根据运行结果得出 Class.forName 加载类时将类进了初始化,再看Classloader。

public class Main {
    public static void main(String[] args) throws ClassNotFoundException {
      ClassLoader.getSystemClassLoader().loadClass("com.hxl.Test");
    }
}

而 ClassLoader 的 loadClass 并没有对类进行初始化,只是把类加载到了虚拟机中。

最后在说说JDBC,我们使用 JDBC 时通常是使用 Class.forName() 方法来加载数据库连接驱动。这是因为在 JDBC 规范中明确要求 Driver(数据库驱动)类必须向 DriverManager 注册自己。

以 MySQL 的驱动为例,我们看到 Driver 注册到 DriverManager 中的操作写在了静态代码块中,这就是为什么在写 JDBC 时使用 Class.forName() 的原因了。

public class Driver extends NonRegisteringDriver implements java.sql.Driver {
    //
    // Register ourselves with the DriverManager
    //
    static {
        try {
            java.sql.DriverManager.registerDriver(new Driver());
        } catch (SQLException E) {
            throw new RuntimeException("Can't register driver!");
        }
    }
    /**
     * Construct a new driver and register it with DriverManager
     * 
     * @throws SQLException
     *             if a database error occurs.
     */
    public Driver() throws SQLException {
        // Required for Class.forName().newInstance()
    }
}



版权声明:本文为HouXinLin_CSDN原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。