浅谈java里Class.forName

本文详细解析了Java中的Class.forName()方法,包括其作用、使用场景及与new关键字的区别。着重讨论了newInstance()方法的使用与JDBC加载驱动类的过程,解释了Bridge模式下DriverManager与Driver之间的关系,以及Driver类如何自动完成注册过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

我们在API文档下从java.lang包内找得Class类的forName()方法:
static Class<?> forName(String className )【常用】
    返回与带有给定字符串名的类或接口相关的Class对象
  用法与【Class.forName(className, true, currentLoader)】相同
static Class<?> forName(String name,boolean intialize,ClassLoader loader)
    使用给定的类加载器,返回与带有给定字符串名的类或接口相关联的Class对象。
Class.forName的作用是什么:

   首先你要明白在java里面任何class都要装载在虚拟机上才能运行。而Class.forName就是装载类用的,new 也可以装载类用的,但与new有区别。通过给定一个字符串变量(它代表一个类的包名和类名)实例化其相关的对象:

  className (className)Class.forName("package.className").newInstance(); 

  className new className(); 
两者是一样的效果。 

Class.forName("package.className") 返回的是一个类,其作用是要求JVM查找并加载指定的类,也就是说JVM会执行该类的静态代码段。

Class.forName的应用:

1)动态加载和创建Class 对象,比如想根据用户输入的字符串来创建对象 
 String str 用户输入的字符串 
 Class Class.forName(str); 
 a.newInstance();

     在初始化一个类,生成一个实例的时候,newInstance()方法和new关键字除了一个是方法,一个是关键字外,最主要的区别在于创建对象的方式不一样,前者是使用类加载机制,后者是创建一个新类。那么为什么会有两种创建对象方式?这主要考虑到软件的可伸缩、可扩展和可重用等软件设计思想。 

2)Java中工厂模式经常使用newInstance()方法来创建对象。 例如: 
class Class.forName(“Example”); 
factory (ExampleInterface)c.newInstance(); 
其中ExampleInterface是Example的接口,可以写成如下形式: 
String className "Example"; 
class Class.forName(className); 
factory (ExampleInterface)c.newInstance(); 
进一步可以写成如下形式: 
String className readfromXMlConfig;//从xml 配置文件中获得字符串 
class Class.forName(className); 
factory (ExampleInterface)c.newInstance(); 

   上面代码已经不存在Example的类名称,它的优点是,无论Example类怎么变化,上述代码不变,甚至可以更换Example的兄弟类Example2 Example3 Example4……,只要他们继承ExampleInterface就可以。 

   从JVM的角度看,我们使用关键字new创建一个类的时候,这个类可以没有被加载。但是使用newInstance()方法的时候,就必须保证:1、这个类已经加载;2、这个类已经连接了。而完成上面两个步骤的正是Class的静态方法forName()所完成的,这个静态方法调用了启动类加载器,即加载java API的那个加载器。 

    现在可以看出,newInstance()实际上是把new这个方式分解为两步,即首先调用Class加载方法加载某个类,然后实例化。 这样分步的好处是显而易见的。我们可以在调用class的静态加载方法forName时获得更好的灵活性,提供给了一种降耦的手段。 

最后用最简单的描述来区分new关键字和newInstance()方法的区别: 
newInstance: 弱类型。低效率。只能调用无参构造。 
new: 强类型。相对高效。能调用任何public构造。

3)使用JDBC时(这部分内容来自http://dustin.iteye.com/blog/44291)

  Class.forName("com.mysql.jdbc.Driver");   

  1. String url "jdbc:mysql://127.0.0.1/test?useUnicode=true&characterEncoding=utf-8";   
  2. String user "";   
  3. String psw "";   
  4. Connection con DriverManager.getConnection(url,user,psw);  


    为什么说很自然呢,因为无论是网上还是书本教程上得例子都是这样的,而且程序也确实正常运行了,于是大家也就心安理得的找葫芦画瓢下去了。
    一定要有这一句吗?不是的,我们完全可以用这样一句代替它:

java 代码
  1. com.mysql.jdbc.Driver driver new com.mysql.jdbc.Driver();   
  2. //or:   
  3. //new com.mysql.jdbc.Driver();   
  4. String url "jdbc:mysql://127.0.0.1/test?useUnicode=true&characterEncoding=utf-8";   
  5. String user "";   
  6. String psw "";   
  7. Connection con DriverManager.getConnection(url,user,psw);   
  8.   


 

    大家可能都看出个大概来了,我们只需要在调用DriverManager的getConnection方法之前,保证相应的Driver类已经被加载到jvm中,并且完成了类的初始化工作就行了,而具体是怎样实现这个功能却是没有讲究的。上面两种方法都可以实现这个功能,因此程序可以正常运行。注意了,如果我们进行如下操作,程序是不能正常运行的,因为这样仅仅使Driver类被装载到jvm中,却没有进行相应的初始化工作。

java 代码
  1. com.mysql.jdbc.Driver driver null;   
  2. //or:   
  3. ClassLoader cl new ClassLoader();   
  4. cl.loadClass("com.mysql.jdbc.Driver");  


     我们都知道JDBC是使用Bridge模式进行设计的,DriverManager就是其中的Abstraction,java.sql.Driver是Implementor,com.mysql.jdbc.Driver是Implementor的一个具体实现(请参考GOF的Bridge模式的描述)。大家注意了,前一个Driver是一个接口,后者却是一个类,它实现了前面的Driver接口。
     Bridge模式中,Abstraction(DriverManager)是要拥有一个Implementor(Driver)的引用的,但是我们在使用过程中,并没有将Driver对象注册到DriverManager中去啊,这是怎么回事呢?jdk文档对Driver的描述中有这么一句:
     When a Driver class is loaded, it should create an instance of itself and register it with the DriverManager
哦,原来是com.mysql.jdbc.Driver在装载完后自动帮我们完成了这一步骤。源代码是这样的:

java 代码
  1. package com.mysql.jdbc   
  2.   
  3. public class Driver extends NonRegisteringDriver implements java.sql.Driver {   
  4.  // Static fields/initializers   
  5.  // --------------------------------------------- //   
  6.  // Register ourselves with the DriverManager   
  7.  //   
  8.  static {   
  9.     ry {   
  10.               java.sql.DriverManager.registerDriver(new Driver());   
  11.           catch (SQLException E) {   
  12.               throw new RuntimeException("Can't register driver!");   
  13.           }   
  14.   }   
  15. // Constructors   
  16.  // -----------------------------------------------------------   
  17.   
  18.  public Driver() throws SQLException {   
  19.      // Required for Class.forName().newInstance()   
  20.  }   
  21.  

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值