浅谈java里Class.forName

来源:互联网 发布:搞笑淘宝客服对话截图 编辑:程序博客网 时间:2024/04/29 11:32
我们在API文档下从java.lang包内找得Class类的forName()方法:
staticClass<?> forName(StringclassName)【常用】
  返回与带有给定字符串名的类或接口相关的Class对象
  用法与【Class.forName(className,true, currentLoader)】相同
staticClass<?> forName(String name,booleanintialize,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 instanceof 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.  

原创粉丝点击