Java如何基于反射機制獲取不同的類
什么是反射機制
Java的反射(reflection)機制是指在程序的運行狀態(tài)中,可以構(gòu)造任意一個類的對象,可以了解任意一個對象所屬的類,可以了解任意一個類的成員變量和方法,可以調(diào)用任意一個對象的屬性和方法。這種動態(tài)獲取程序信息以及動態(tài)調(diào)用對象的功能稱為Java語言的反射機制。反射被視為動態(tài)語言的關(guān)鍵(在運行階段可以改變其結(jié)構(gòu))
特點:
1、極大的提高了程序的靈活性和擴展性,降低模塊的耦合性,提高自身的適應(yīng)能力
2、通過反射機制可以讓程序創(chuàng)建和控制任何類的對象,無需提前硬編碼目標(biāo)類
3、能夠在運行時構(gòu)造一個類的對象、判斷一個類所具有的成員變量和方法、調(diào)用一個對象的方法(包括 private),可能會有安全隱患
4、是構(gòu)建框架技術(shù)的基礎(chǔ)所在,使用反射可以避免將代碼寫死在框架中
5、包含動態(tài)類型,而 JVM 不能對動態(tài)代碼優(yōu)化,所以效率會比較低
在Java中可以通過Reflection APIs在運行時動態(tài)生成class實體,通過Constructor創(chuàng)建類實例、Field訪問成員屬性、Method調(diào)用方法
Class類
在Java中,每個 class 都有一個相應(yīng)的 Class 對象。在編譯完成后,在生成的.class文件中,會產(chǎn)生一個Class對象,用于表示這個類的類型信息。
特點:
1、由系統(tǒng)創(chuàng)建Class類
2、同一個類的實例指向同一個Class對象
3、通過Class可以完整的得到一個類所有被加載的結(jié)構(gòu)
4、Class類是Java反射的基礎(chǔ)
獲得Class類的方法:
public class MyReflect {
public static void main(String[] args) throws ClassNotFoundException {
Teacher t = new Teacher();
// 1、通過對象獲得
Class c1 = t.getClass();
// 2、forName + 全限定名 獲得
Class c2 = Class.forName("reflect.Teacher");
// 3、通過類名獲得
Class c3 = Teacher.class;
// 4、基本數(shù)據(jù)類型的Type屬性獲得
Class c4 = Double.TYPE;
// 5、通過子類的Class對象獲得
Class c5 = c1.getSuperclass();
}
}
class Person {
String name;
public Person() {
}
}
class Teacher extends Person {
public Teacher() {
}
}

getXXX為獲取 public 構(gòu)造器、屬性、方法
getDeclaredXXX 獲取任意構(gòu)造器、屬性、方法(包括private)
setAccessible方法避開權(quán)限檢測,用以訪問權(quán)限不足的成員
以下不在贅述
準(zhǔn)備:Person類以及Tercher子類
package reflect;
class Person {
public String name;
private int age;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public Person() {
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
public String greet(String name) {
return "hello " + name + ".";
}
}
class Teacher extends Person {
}
Constructor類
java.lang.reflect.Constructor類是java.lang.reflect.Executable類的直接子類,用于表示類的構(gòu)造方法。通過Class對象的 getConstructors() 方法可以獲得當(dāng)前運行時類的構(gòu)造方法
public class MyConstructor {
public static void main(String[] args) throws Exception {
Class<Person> c = (Class<Person>) Class.forName("reflect.Person");
// Object person2 = c.newInstance(); //@Deprecated
// System.out.println(person2);
Constructor<Person> constructor = c.getConstructor();
Person person = constructor.newInstance();
System.out.println(person);
// 獲取 public 有參構(gòu)造方法
Constructor<Person> constructor1 = c.getConstructor(String.class, int.class);
Person person1 = constructor1.newInstance("JL", 18);
System.out.println(person1);
// 獲取構(gòu)造方法
Constructor<Person> constructor2 = c.getDeclaredConstructor(String.class, int.class);
constructor2.setAccessible(true);
Person person2 = constructor1.newInstance("JL", 18);
System.out.println(person2.name);
}
}
Field類
java.lang.reflect.Field類用于封裝成員變量信息,調(diào)用Class對象的 getField() 或 getFields()等方法可以獲得當(dāng)前運行時類的成員變量
public class MyField {
public static void main(String[] args) throws Exception {
Class<?> c = Class.forName("reflect.Person");
// 在Person類中,無參的構(gòu)造方法不能省略
Object p = c.getConstructor().newInstance();
// 獲取 public 成員變量
Field[] fields = c.getFields();
// 獲取所有成員變量(包括 private)
Field[] allFields = c.getDeclaredFields();
//獲取指定名稱public變量
Field f1 = c.getField("name");
// 成員變量set、get
f1.set(p, "JL");
System.out.println(f1.get(p));
Field f2 = c.getDeclaredField("age"); // private int age
// 忽略權(quán)限檢查,使得 private 成員可以被訪問
f2.setAccessible(true);
f2.set(p, 18);
System.out.println(f2.get(p));
}
}
Method類
java.lang.reflect.Method類是java.lang.reflect.Executable類的直接子類,用于封裝成員方法的信息,調(diào)用Class對象的 getMethod() 方法或 getMethods() 方法可以獲得當(dāng)前運行時類的指定方法或所有方法
public class MyMethod {
public static void main(String[] args) throws Exception {
Class<Person> c = (Class<Person>) Class.forName("reflect.Person");
Person p = c.getConstructor().newInstance();
// 獲取所有 public 方法
Method[] methods = c.getMethods();
// 獲取 greet 方法
Method method1 = c.getMethod("greet", String.class);
//執(zhí)行 greet 方法
System.out.println(method1.invoke(p, "JL"));
}
}
0
PS:主要簡單說明反射的使用,關(guān)于類加載機制,性能分析等沒有說明
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
SpringMVC用XML方式實現(xiàn)AOP的方法示例
這篇文章主要介紹了SpringMVC用XML方式實現(xiàn)AOP的方法示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04
springboot啟動腳本start.sh和停止腳本 stop.sh的詳細(xì)教程
這篇文章主要介紹了springboot啟動腳本start.sh和停止腳本 stop.sh的詳細(xì)教程,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-08-08
SpringBoot實現(xiàn)mysql與clickhouse多數(shù)據(jù)源的項目實踐
本文主要介紹了SpringBoot實現(xiàn)mysql與clickhouse多數(shù)據(jù)源的項目實踐,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-11-11
Springboot+Netty+Websocket實現(xiàn)消息推送實例
這篇文章主要介紹了Springboot+Netty+Websocket實現(xiàn)消息推送實例,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-02-02
淺析打開eclipse出現(xiàn)Incompatible JVM的解決方法
本篇文章是對打開eclipse出現(xiàn)Incompatible JVM的解決方法進行了詳細(xì)的分析介紹,需要的朋友參考下2013-07-07
Spring Cloud學(xué)習(xí)教程之Zuul統(tǒng)一異常處理與回退
Spring Cloud Zuul對異常的處理整體來說還是比較方便的,流程也比較清晰,下面這篇文章主要給大家介紹了關(guān)于Spring Cloud學(xué)習(xí)教程之Zuul統(tǒng)一異常處理與回退的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2018-04-04
Java Web基于Session的登錄實現(xiàn)方法
這篇文章主要介紹了Java Web基于Session的登錄實現(xiàn)方法,涉及Java針對session的操作及表單提交與驗證技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-10-10

