0104. Maximum Depth of Binary Tree (E)
Java 反射機制概述
Reflection(反射)是被視為動態語言的關鍵,反射機制允許程式在執行期藉助於Reflection API取得任何類的內部資訊,並能直接操作任意物件的內部屬性及方法。
載入完類之後,在堆記憶體的方法區中就產生了一個Class型別的物件(一個類只有一個Class物件),這個物件就包含了完整的類的結構資訊。我們可以通過這個物件看到類的結構。這個物件就像一面鏡子,透過這個鏡子看到類的結構,所以,我們形象的稱之為: 反射。
補充:動態語言 vs 靜態語言
1 、動態語言
是一類在執行時可以改變其結構的語言:例如新的函式、物件、甚至程式碼可以被引進,已有的函式可以被刪除或是其他結構上的變化。通俗點說就是 在執行時程式碼可以根據某些條件改變自身結構。
主要動態語言:Object-C、C#、JavaScript、PHP、Python、Erlang。
2 、靜態語言
與動態語言相對應的,執行時結構不可變的語言就是靜態語言。如Java、C、C++。
Java不是動態語言,但Java可以稱之為“準動態語言”。即Java有一定的動態性,我們可以利用反射機制、位元組碼操作獲得類似動態語言的特性。Java的動態性讓程式設計的時候更加靈活!
理解Class 類並獲取Class
在Object類中定義了以下的方法,此方法將被所有子類繼承:
public final Class getClass()
以上的方法返回值的型別是一個Class類,此類是Java反射的源頭,實際上所謂反射從程式的執行結果來看也很好理解,即:可以通過物件反射求出類的名稱。
物件照鏡子後可以得到的資訊:某個類的屬性、方法和構造器、某個類到底實現了哪些介面。對於每個類而言,JRE 都為其保留一個不變的 Class 型別的物件。一個 Class 物件包含了特定某個結構(class/interface/enum/annotation/primitive type/void/[])的有關資訊。
Class本身也是一個類
Class 物件只能由系統建立物件
一個載入的類在 JVM 中只會有一個Class例項
一個Class物件對應的是一個載入到JVM中的一個.class檔案
每個類的例項都會記得自己是由哪個 Class 例項所生成
通過Class可以完整地得到一個類中的所有被載入的結構
Class類是Reflection的根源,針對任何你想動態載入、執行的類,唯有先獲得相應的Class物件
Class 類的常用方法
獲取Class 類的例項( 四種方法)
反射機制的相關類
與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() | 獲得當前類實現的類或是介面 |
- 獲得類中屬性相關的方法
方法 | 用途 |
---|---|
getField(String name) | 獲得某個公有的屬性物件 |
getFields() | 獲得所有公有的屬性物件 |
getDeclaredField(String name) | 獲得某個屬性物件 |
getDeclaredFields() | 獲得所有屬性物件 |
- 獲得類中註解相關的方法
方法 | 用途 |
---|---|
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) | 根據傳遞的引數建立類的物件 |
示例
- 被反射類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();
}
}
}