newInstance和new的區別(good)

滄海一滴發表於2014-04-07


從JVM 的角度看,我們使用關鍵字new建立一個類的時候,這個類可以沒有被載入。
但是使用newInstance()方法的時候,就必須保證:
1、這個 類已經載入
2、這個類已經連線了。
而完成上面兩個步驟的正是Class的靜態方法forName()所完成的,這個靜態方法呼叫了啟動類載入器,即載入 java API的那個載入器。

現在可以看出,newInstance()實際上是把new這個方式分解為兩步,即
(1)首先呼叫Class載入方法載入某個類,
(2)然後例項化。 這樣分步的好處是顯而易見的。

我們可以在呼叫class的靜態載入方法forName時獲得更好的靈活性,提供給了一種降耦的手段。

使用JDBC時候,常見的語句如下:

Class.forName("com.mysql.jdbc.Driver");   
String url = "jdbc:mysql://127.0.0.1/test?useUnicode=true&characterEncoding=utf-8";   
String user = "";   
String psw = "";   
Connection con = DriverManager.getConnection(url,user,psw);
 

其中第一句是為了載入資料庫驅動,它也可以寫成:

com.mysql.jdbc.Driver driver = new com.mysql.jdbc.Driver();   
//or:   
//new com.mysql.jdbc.Driver();
 

為了保證相應的Driver類已經被載入到 jvm中,並且完成了類的初始化工作就行了,而具體是怎樣實現這個功能卻是沒有講究的。

裝載的程式碼如下:

package com.mysql.jdbc   
  
public class Driver extends NonRegisteringDriver implements java.sql.Driver {   
// ~ Static fields/initializers   
// --------------------------------------------- //   
// Register ourselves with the DriverManager   
//   
static {   
     t ry {   
               java.sql.DriverManager.registerDriver(new Driver());   
           } catch (SQLException E) {   
              throw new RuntimeException("Can't register driver!");   
           }   
   }   
// ~ Constructors   
// -----------------------------------------------------------   
/**
   * 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()   
}   
}
 

在初始化一個類,生成一個例項的時候, newInstance()方法和new關鍵 除了一個是方法,一個是關鍵字外,

 

最主要區別在於

建立物件的方式不一樣,前者是使用類載入機制,後者是建立一個新類。

 

那麼為什麼會有兩種建立物件方式?這主要考慮到軟體的可伸縮、可擴充套件和可重用等軟體設計思想。 

Java中工廠模式經常使用newInstance()方法來建立物件,因此從為什麼要使用工廠模式上可以找到具體答案。 例如: 
class c = Class.forName(“Example”); 
factory = (ExampleInterface)c.newInstance(); 


其中ExampleInterface是Example的介面,可以寫成如下形式: 
String className = "Example"; 
class c = Class.forName(className); 
factory = (ExampleInterface)c.newInstance(); 


進一步可以寫成如下形式: 
String className = readfromXMlConfig;//從xml 配置檔案中獲得字串 
class c = 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構造

以上文字主要摘自http://blog.csdn.net/hepenghit/article/details/5286398

在此主要對newInstance和new的區別做個記錄

http://my.oschina.net/bayuanqian/blog/89627

相關文章