Java類的加載及實例的創建


 

java中class.forName()和classLoader都可用來對類進行加載。
class.forName()前者除了將類的.class文件加載到jvm中之外,還會對類進行解釋,執行類中的static塊。
而classLoader只干一件事情,就是將.class文件加載到jvm中,不會執行static中的內容,只有在newInstance才會去執行static塊。
Class.forName(name, initialize, loader)帶參函數也可控制是否加載static塊。並且只有調用了newInstance()方法采用調用構造函數,創建類的對象

1、Java創建對象的幾種方式

 

Java創建對象的幾種方式(重要):

(1) 用new語句創建對象,這是最常見的創建對象的方法。
(2) 運用反射手段,調用java.lang.Class或者java.lang.reflect.Constructor類的newInstance()實例方法。
(3) 調用對象的clone()方法。
(4) 運用反序列化手段,調用java.io.ObjectInputStream對象的 readObject()方法。

(1)和(2)都會明確的顯式的調用構造函數 ;(3)和(4)也不會調用構造函數。

 

2、Class.forName()介紹

 

在java.lang.Class中,有兩個重載的forName方法,分別是:

  1. static Class<?> forName(String className),該方法等價於Class.forName(className, true, this.getClass().getClassLoader())
  2. static Class<?> forName(String className, boolean initialize,ClassLoader loader),其中3個參數分別表示:className - 所需類的完全限定名,initialize - 是否必須初始化類,loader - 用於加載類的類加載器。

 

forName方法的作用就是:


使用給定的類加載器,返回與帶有給定字符串名的類或接口相關聯的 Class 對象。給定一個類或接口的完全限定名,此方法會試圖定位、加載和鏈接該類或接口。指定的類加載器用於加載該類或接口,如果參數loader 為 null,則該類通過引導類加載器加載。只有 initialize 參數為 true且以前未被初始化時,才初始化該類。

 

public class TestClassForName {
 MyTest tt;
 public static void main(String[] args) throws Exception {
   (new TestClassForName()).loadClass();
 }

 public void loadClass() throws Exception {
   // 會執行靜態代碼
   // Class<MyTest> clazz = (Class<MyTest>) Class.forName("MyTest", true, getClass().getClassLoader());
   Class<MyTest> clazz = (Class<MyTest>)Class.forName("MyTest");
   // 執行構造函數
   tt = (MyTest)clazz.newInstance();
   // 執行實例方法
   tt.test();

 }
}

class MyTest {
 static {
   System.out.println("類的靜態初始化塊");
 }

 public MyTest() {
   System.out.println("實例化類");
 }
 
 public void test(){
   System.out.println("實例方法");
 }
}
類的靜態初始化塊
實例化類
實例方法

 

3、Class.forName()與new的區別

 

在java里面任何class都要裝載在虛擬機上才能運行。Class.forName(xxx.xx.xx) 返回的是一個類。這句話就是裝載類用的(和new 不一樣,要分清楚)。

給你一個字符串變量,它代表一個類的包名和類名,你怎么實例化它?只有用你提到的這個方法了,不過要再加一點。 

A a = (A)Class.forName("pacage.A").newInstance();
A a = new A();

如上的兩行代碼是等價的。

jvm會執行靜態代碼段,靜態代碼是和class綁定的,class裝載成功就表示執行了你的靜態代碼了。而且以后不會再走這段靜態代碼了。Class.forName(xxx.xx.xx) 返回的是一個類,作用是要求JVM查找並加載指定的類,也就是說JVM會執行該類的靜態代碼段

動態加載和創建Class 對象,比如想根據用戶輸入的字符串來創建對象 

String str = 用戶輸入的字符串 
Class t = Class.forName(str); 
t.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構造。

 
 
 
 
 
 
 
 
 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM