1. 程式人生 > 程式設計 >詳解Java高階特性之反射

詳解Java高階特性之反射

定義

JAVA反射機制是在執行狀態中,對於任意一個類,都能夠知道這個類的所有屬性和方法;對於任意一個物件,都能夠呼叫它的任意方法和屬性;這種動態獲取資訊以及動態呼叫物件方法的功能稱為java語言的反射機制。

用途

在日常的第三方應用開發過程中,經常會遇到某個類的某個成員變數、方法或是屬性是私有的或是隻對系統應用開放,這時候就可以利用Java的反射機制通過反射來獲取所需的私有成員或是方法。當然,也不是所有的都適合反射,之前就遇到一個案例,通過反射得到的結果與預期不符。閱讀原始碼發現,經過層層呼叫後在最終返回結果的地方對應用的許可權進行了校驗,對於沒有許可權的應用返回值是沒有意義的預設值,否則返回實際值起到保護使用者的隱私目的。

反射機制的相關類

與Java反射相關的類如下:

類名 用途
Class類 代表類的實體,在執行的Java應用程式中表示類和介面
Field類 代表類的成員變數(成員變數也稱為類的屬性)
Method類 代表類的方法
Constructor類 代表類的構造方法

Class類

Class代表類的實體,在執行的Java應用程式中表示類和介面。在這個類中提供了很多有用的方法,這裡對他們簡單的分類介紹。

獲得類相關的方法

方法 用途
asSubclass(Class<U> clazz) 把傳遞的類的物件轉換成代表其子類的物件
Cast 把物件轉換成代表類或是介面的物件
getClassLoader() 獲得類的載入器
getClasses() 返回一個數組,陣列中包含該類中所有公共類和介面類的物件
getDeclaredClasses() 返回一個數組,陣列中包含該類中所有類和介面類的物件
forName(String className) 根據類名返回類的物件
getName() 獲得類的完整路徑名字
newInstance() 建立類的例項
getPackage() 獲得類的包
getSimpleName() 獲得類的名字
getSuperclass() 獲得當前類繼承的父類的名字
getInterfaces() 獲得當前類實現的類或是介面

獲得類中屬性相關的方法

方法 用途
getAnnotation(Class<A> annotationClass) 返回該類中與引數型別匹配的公有註解物件
getAnnotations() 返回該類所有的公有註解物件
getDeclaredAnnotation(Class<A> annotationClass) 返回該類中與引數型別匹配的所有註解物件
getDeclaredAnnotations() 返回該類所有的註解物件

獲得類中構造器相關的方法

方法 用途
getConstructor(Class...<?> parameterTypes) 獲得該類中與引數型別匹配的公有構造方法
getConstructors() 獲得該類的所有公有構造方法
getDeclaredConstructor(Class...<?> parameterTypes) 獲得該類中與引數型別匹配的構造方法
getDeclaredConstructors() 獲得該類所有構造方法

獲得類中方法相關的方法


方法 用途
getMethod(String name,Class...<?> parameterTypes) 獲得該類某個公有的方法
getMethods() 獲得該類所有公有的方法
getDeclaredMethod(String name,Class...<?> parameterTypes) 獲得該類某個方法
getDeclaredMethods() 獲得該類所有方法

類中其他重要的方法

方法 用途
isAnnotation() 如果是註解型別則返回true
isAnnotationPresent(Class<? extends Annotation> annotationClass) 如果是指定型別註解型別則返回true
isAnonymousClass() 如果是匿名類則返回true
isArray() 如果是一個數組類則返回true
isEnum() 如果是列舉類則返回true
isInstance(Object obj) 如果obj是該類的例項則返回true
isInterface() 如果是介面類則返回true
isLocalClass() 如果是區域性類則返回true
isMemberClass() 如果是內部類則返回true

Field類

Field代表類的成員變數(成員變數也稱為類的屬性)。

方法 用途
equals(Object obj) 屬性與obj相等則返回true
get(Object obj) 獲得obj中對應的屬性值
set(Object obj,Object value) 設定obj中對應屬性值

Method類

Method代表類的方法。

方法 用途
invoke(Object obj,Object... args) 傳遞object物件及引數呼叫該物件對應的方法

Constructor類

Constructor代表類的構造方法。

方法 用途
newInstance(Object... initargs) 根據傳遞的引數建立類的物件

示例

為了演示反射的使用,首先構造一個與書籍相關的model——Book.java,然後通過反射方法示例建立物件、反射私有構造方法、反射私有屬性、反射私有方法,最後給出兩個比較複雜的反射示例——獲得當前ZenMode和關機Shutdown。

被反射類Book.java

public class Book{
  private final static String TAG = "BookTag";
 
  private String name;
  private String author;
 
  @Override
  public String toString() {
    return "Book{" +
        "name='" + name + '\'' +
        ",author='" + author + '\'' +
        '}';
  }
 
  public Book() {
  }
 
  private Book(String name,String author) {
    this.name = name;
    this.author = author;
  }
 
  public String getName() {
    return name;
  }
 
  public void setName(String name) {
    this.name = name;
  }
 
  public String getAuthor() {
    return author;
  }
 
  public void setAuthor(String author) {
    this.author = author;
  }
 
  private String declaredMethod(int index) {
    String string = null;
    switch (index) {
      case 0:
        string = "I am declaredMethod 1 !";
        break;
      case 1:
        string = "I am declaredMethod 2 !";
        break;
      default:
        string = "I am declaredMethod 1 !";
    }
 
    return string;
  }
}

反射邏輯封裝在ReflectClass.java  

public class ReflectClass {
  private final static String TAG = "peter.log.ReflectClass";
 
  // 建立物件
  public static void reflectNewInstance() {
    try {
      Class<?> classBook = Class.forName("com.android.peter.reflectdemo.Book");
      Object objectBook = classBook.newInstance();
      Book book = (Book) objectBook;
      book.setName("Android進階之光");
      book.setAuthor("劉望舒");
      Log.d(TAG,"reflectNewInstance book = " + book.toString());
    } catch (Exception ex) {
      ex.printStackTrace();
    }
  }
 
  // 反射私有的構造方法
  public static void reflectPrivateConstructor() {
    try {
      Class<?> classBook = Class.forName("com.android.peter.reflectdemo.Book");
      Constructor<?> declaredConstructorBook = classBook.getDeclaredConstructor(String.class,String.class);
      declaredConstructorBook.setAccessible(true);
      Object objectBook = declaredConstructorBook.newInstance("Android開發藝術探索","任玉剛");
      Book book = (Book) objectBook;
      Log.d(TAG,"reflectPrivateConstructor book = " + book.toString());
    } catch (Exception ex) {
      ex.printStackTrace();
    }
  }
 
  // 反射私有屬性
  public static void reflectPrivateField() {
    try {
      Class<?> classBook = Class.forName("com.android.peter.reflectdemo.Book");
      Object objectBook = classBook.newInstance();
      Field fieldTag = classBook.getDeclaredField("TAG");
      fieldTag.setAccessible(true);
      String tag = (String) fieldTag.get(objectBook);
      Log.d(TAG,"reflectPrivateField tag = " + tag);
    } catch (Exception ex) {
      ex.printStackTrace();
    }
  }
 
  // 反射私有方法
  public static void reflectPrivateMethod() {
    try {
      Class<?> classBook = Class.forName("com.android.peter.reflectdemo.Book");
      Method methodBook = classBook.getDeclaredMethod("declaredMethod",int.class);
      methodBook.setAccessible(true);
      Object objectBook = classBook.newInstance();
      String string = (String) methodBook.invoke(objectBook,0);
 
      Log.d(TAG,"reflectPrivateMethod string = " + string);
    } catch (Exception ex) {
      ex.printStackTrace();
    }
  }
 
  // 獲得系統Zenmode值
  public static int getZenMode() {
    int zenMode = -1;
    try {
      Class<?> cServiceManager = Class.forName("android.os.ServiceManager");
      Method mGetService = cServiceManager.getMethod("getService",String.class);
      Object oNotificationManagerService = mGetService.invoke(null,Context.NOTIFICATION_SERVICE);
      Class<?> cINotificationManagerStub = Class.forName("android.app.INotificationManager$Stub");
      Method mAsInterface = cINotificationManagerStub.getMethod("asInterface",IBinder.class);
      Object oINotificationManager = mAsInterface.invoke(null,oNotificationManagerService);
      Method mGetZenMode = cINotificationManagerStub.getMethod("getZenMode");
      zenMode = (int) mGetZenMode.invoke(oINotificationManager);
    } catch (Exception ex) {
      ex.printStackTrace();
    }
 
    return zenMode;
  }
 
  // 關閉手機
  public static void shutDown() {
    try {
      Class<?> cServiceManager = Class.forName("android.os.ServiceManager");
      Method mGetService = cServiceManager.getMethod("getService",String.class);
      Object oPowerManagerService = mGetService.invoke(null,Context.POWER_SERVICE);
      Class<?> cIPowerManagerStub = Class.forName("android.os.IPowerManager$Stub");
      Method mShutdown = cIPowerManagerStub.getMethod("shutdown",boolean.class,String.class,boolean.class);
      Method mAsInterface = cIPowerManagerStub.getMethod("asInterface",IBinder.class);
      Object oIPowerManager = mAsInterface.invoke(null,oPowerManagerService);
      mShutdown.invoke(oIPowerManager,true,null,true);
 
    } catch (Exception ex) {
      ex.printStackTrace();
    }
  }
 
  public static void shutdownOrReboot(final boolean shutdown,final boolean confirm) {
    try {
      Class<?> ServiceManager = Class.forName("android.os.ServiceManager");
      // 獲得ServiceManager的getService方法
      Method getService = ServiceManager.getMethod("getService",java.lang.String.class);
      // 呼叫getService獲取RemoteService
      Object oRemoteService = getService.invoke(null,Context.POWER_SERVICE);
      // 獲得IPowerManager.Stub類
      Class<?> cStub = Class.forName("android.os.IPowerManager$Stub");
      // 獲得asInterface方法
      Method asInterface = cStub.getMethod("asInterface",android.os.IBinder.class);
      // 呼叫asInterface方法獲取IPowerManager物件
      Object oIPowerManager = asInterface.invoke(null,oRemoteService);
      if (shutdown) {
        // 獲得shutdown()方法
        Method shutdownMethod = oIPowerManager.getClass().getMethod(
            "shutdown",boolean.class);
        // 呼叫shutdown()方法
        shutdownMethod.invoke(oIPowerManager,confirm,false);
      } else {
        // 獲得reboot()方法
        Method rebootMethod = oIPowerManager.getClass().getMethod("reboot",boolean.class);
        // 呼叫reboot()方法
        rebootMethod.invoke(oIPowerManager,false);
      }
    } catch (Exception e) {
      e.printStackTrace();
    }
  }
}

呼叫相應反射邏輯方法 

try {
      // 建立物件
      ReflectClass.reflectNewInstance();
 
      // 反射私有的構造方法
      ReflectClass.reflectPrivateConstructor();
 
      // 反射私有屬性
      ReflectClass.reflectPrivateField();
 
      // 反射私有方法
      ReflectClass.reflectPrivateMethod();
    } catch (Exception ex) {
      ex.printStackTrace();
    }
 
    Log.d(TAG," zenmode = " + ReflectClass.getZenMode());

Log輸出結果如下:

08-27 15:11:37.999 11987-11987/com.android.peter.reflectdemo D/peter.log.ReflectClass: reflectNewInstance book = Book{name='Android進階之光',author='劉望舒'}
08-27 15:11:38.000 11987-11987/com.android.peter.reflectdemo D/peter.log.ReflectClass: reflectPrivateConstructor book = Book{name='Android開發藝術探索',author='任玉剛'}
08-27 15:11:38.000 11987-11987/com.android.peter.reflectdemo D/peter.log.ReflectClass: reflectPrivateField tag = BookTag
08-27 15:11:38.000 11987-11987/com.android.peter.reflectdemo D/peter.log.ReflectClass: reflectPrivateMethod string = I am declaredMethod 1 !
08-27 15:11:38.004 11987-11987/com.android.peter.reflectdemo D/peter.log.ReflectDemo: zenmode = 0

以上就是詳解Java高階特性之反射的詳細內容,更多關於JAVA高階特性反射的資料請關注我們其它相關文章!