Java反射機制詳解上篇

1反射機制是什麼

反射機制是在運行狀態中,對於任意一個類,都可以知道這個類的全部屬性和方法;對於任意一個對象,都可以調用它的任意一個方法和屬性;這種動態獲取的信息以及動態調用對象的方法的功能稱爲java語言的反射機制。java


在面向對象的世界裏,萬事萬物皆對象.在java語言裏,靜態的成員,普通數據類型是否是對象呢?
編程

類又是誰的對象呢?
設計模式

首先類是對象,類是java.lang.Class類的實例對象.
app

新建立一個Foo類
eclipse

Foo這個類也是一個實例對象,是Class類的實例對象,這個對象在官網被稱爲(class type).ide

反射是java程序開發語言的特性之一,它容許運行中的java程序獲取自身的信息,而且能夠操做類或者對象內部的屬性.學習

反射的核心是JVM在運行時才動態加載類或調用方法/訪問屬性,它不須要事先(寫代碼的時候或編譯期)知道運行對象是誰。ui

瞭解反射其實須要瞭解JVM,不過通常的資料不會在這一部分講到JVM,畢竟學習也是要從淺入深的.spa


2反射機制能作什麼設計


反射機制主要提供瞭如下功能: 

  • 在運行時判斷任意一個對象所屬的類;

  • 在運行時構造任意一個類的對象;

  • 在運行時判斷任意一個類所具備的成員變量和方法;

  • 在運行時調用任意一個對象的方法;

  • 生成動態代理。

注意;是運行時獲取而不是編譯時獲取.其實不少時候咱們直接用eclipse寫代碼忽略了編譯的過程

在Eclipse,當咱們輸入一個點的時候(好比 a.)   編譯器就會自動列出它的屬性和方法,這裏就會用到反射


3反射機制的相關API (在這裏只說反射機制五種功能的前兩種)

java的反射機制的實現要藉助於4個類: class,Constructor,Field,Method;

經過一個對象得到完整的包名和類名

package cn.xins08.boke;

public class TestReflect {

    public static void main(String[] args) {
        TestReflect testReflect = new TestReflect();
        System.out.println(testReflect.getClass().getName());
        // 結果:cn.xins08.boke.TestReflect

    }

}


實例化Class對象:

實現反射機制獲取類有三種方法:


package cn.xins08.boke;

public class TestReflect {
    public static void main(String[] args) throws Exception {

        Class<?> class1 = null;
        Class<?> class2 = null;
        Class<?> class3 = null;
        //第一種方式(在JDBC開發中經常使用此方法加載數據驅動)
        class1 = Class.forName("cn.xins08.boke.TestReflect");
      //第三種方式:java中任何一個對象都有getClass方法
        class2 = new TestReflect().getClass();
        
        //第三種方式java中任何一個對象都有class屬性
        class3 = TestReflect.class;
        System.out.println("類名稱: " + class1.getName());
        System.out.println("類名稱: " + class2.getName());
        System.out.println("類名稱: " + class3.getName());
        // 打印結果:
        /*
         * 類名稱: cn.xins08.boke.TestReflect 類名稱: cn.xins08.boke.TestReflect 類名稱:
         * cn.xins08.boke.TestReflect
         */
    }
}


獲取一個對象的父類與實現的接口:

package cn.xins08.boke;

import java.io.Serializable;

public class TestReflect implements Serializable {
     private static final long serialVersionUID = -2862585049955236662L;
    public static void main(String[] args) throws Exception {
         
            
                Class<?> clazz = Class.forName("cn.xins08.boke.TestReflect");
                // 取得父類
                Class<?> parentClass = clazz.getSuperclass();
                System.out.println("clazz的父類爲:" + parentClass.getName());
                // clazz的父類爲: java.lang.Object
                // 獲取全部的接口
                Class<?> intes[] = clazz.getInterfaces();
                System.out.println("clazz實現的接口有:");
                for (int i = 0; i < intes.length; i++) {
                    System.out.println((i + 1) + ":" + intes[i].getName());
                }
                // clazz實現的接口有:
                // 1:java.io.Serializable
        
    }
}


建立實例:

經過反射來生成對象主要有兩種方式:

(1)使用Class對象的newInstance()方法來建立Class對象對應類的實例。

Class<?> c = String.class;
Object str = c.newInstance();

    初始化一個類,生成一個實例的時候,new與newInstance() 有什麼區別?

     區別在於建立對象的方式不同,前者是使用類加載機制,那麼爲何會有兩種建立對象方式?這個就要從可伸縮、可擴展,可重用等軟件思想上解釋了。

newInstance: 弱類型。低效率。只能調用無參構造。
new: 強類型。相對高效。能調用任何public構造。
newInstance()是實現IOC、反射、面對接口編程 和 依賴倒置 等技術方法的必然選擇,new 只能實現具體類的實例化,不適合於接口編程。


(2)先經過Class對象獲取指定的Constructor對象,再調用Constructor對象的newInstance()方法來建立實例。這種方法能夠用指定的構造器構造類的實例。

// 獲取String所對應的Class對象
        Class<?> c = String.class;
        // 獲取String類帶一個String參數的構造器
        Constructor constructor = c.getConstructor(String.class);
        // 根據構造器建立實例
        Object obj = constructor.newInstance("23333");
        System.out.println(obj);



注意事項:反射會額外的消耗必定的系統資源,若是不須要動態的建立一個對象,那麼就不須要用反射


在文章的最後咱們討論下工廠模式:

package cn.xins08.boke;

public interface Fruit {
public void eat();

}
package cn.xins08.boke;

public class Apple implements Fruit{
    public void eat(){
        System.out.println("吃蘋果");
    }

    

}
package cn.xins08.boke;

public class Factory {
public static Fruit getInstance(String className){
    if("apple".equals(className)){
        return new Apple();
    }
    return null;
}
}
package cn.xins08.boke;

public class FactoryDemo {

    public static void main(String[] args) {
        Fruit f = Factory.getInstance("apple");
        f.eat();

    }

}
//返回的結果是"吃蘋果"

這種是一個最簡單的工廠設計模式,可是有一個很大的問題是,若是如今接口的子類增長了,那麼工廠類確定須要改.而形成這個問題的病因就是new,若是變成反射機制就不同了.反射的實例化對象只須要包.類就能夠了.

package cn.xins08.boke;

public interface Fruit {
public void eat();

}
package cn.xins08.boke;

public class Orange implements Fruit{
    public void eat(){
        System.out.println("吃橘子");
    }

    

}
package cn.xins08.boke;

public class Apple implements Fruit{
    public void eat(){
        System.out.println("吃蘋果");
    }

    }
package cn.xins08.boke;

public class Factory {
public static Fruit getInstance(String className){
   Fruit fruit = null;
try{
   //這裏是Fruit的類類型,作強制類型轉換 
    fruit = (Fruit) Class.forName(className).newInstance() ;
    } catch(Exception e) {
       e.printStackTrace();
   }
       return  f ;
       }
  }



}
package cn.xins08.boke;

public class FactoryDemo {

    public static void main(String[] args) {
        Fruit f = Factory.getInstance("cn.xins08.boke.Apple");
        f.eat();

    }

}



這時候咱們發現即便增長几個接口的子類,工廠類照樣能夠完成對象的實例化操做,能夠對應全部的變化.


    到目前爲止你已經會了最基本的反射使用了,在學習Spring以前先講這些,等學習了Spring的依賴注入,反轉控制以後,相信會對反射有更好的理解.


關於反射如下功能,等寫完後會在本文加連接

  • 在運行時判斷任意一個類所具備的成員變量和方法;

  • 在運行時調用任意一個對象的方法;

  • 生成動態代理。


--java一萬小時成神之路--本文首發與51CTO博客---


近期讀書計劃:

《思考,快與慢》

《浪潮之巔》

java的反射機制一共分爲上下兩篇,反射機制下篇請參考:http://xinsz08.blog.51cto.com/10565212/1947327

相關文章
相關標籤/搜索