您好,登錄后才能下訂單哦!
這篇文章主要介紹了Java基于反射機(jī)制獲取不同類的方法,具有一定借鑒價(jià)值,需要的朋友可以參考下。希望大家閱讀完這篇文章后大有收獲。下面讓小編帶著大家一起了解一下。
什么是反射機(jī)制
Java的反射(reflection)機(jī)制是指在程序的運(yùn)行狀態(tài)中,可以構(gòu)造任意一個(gè)類的對(duì)象,可以了解任意一個(gè)對(duì)象所屬的類,可以了解任意一個(gè)類的成員變量和方法,可以調(diào)用任意一個(gè)對(duì)象的屬性和方法。這種動(dòng)態(tài)獲取程序信息以及動(dòng)態(tài)調(diào)用對(duì)象的功能稱為Java語(yǔ)言的反射機(jī)制。反射被視為動(dòng)態(tài)語(yǔ)言的關(guān)鍵(在運(yùn)行階段可以改變其結(jié)構(gòu))
特點(diǎn):
1、極大的提高了程序的靈活性和擴(kuò)展性,降低模塊的耦合性,提高自身的適應(yīng)能力
2、通過(guò)反射機(jī)制可以讓程序創(chuàng)建和控制任何類的對(duì)象,無(wú)需提前硬編碼目標(biāo)類
3、能夠在運(yùn)行時(shí)構(gòu)造一個(gè)類的對(duì)象、判斷一個(gè)類所具有的成員變量和方法、調(diào)用一個(gè)對(duì)象的方法(包括 private),可能會(huì)有安全隱患
4、是構(gòu)建框架技術(shù)的基礎(chǔ)所在,使用反射可以避免將代碼寫死在框架中
5、包含動(dòng)態(tài)類型,而 JVM 不能對(duì)動(dòng)態(tài)代碼優(yōu)化,所以效率會(huì)比較低
在Java中可以通過(guò)Reflection APIs在運(yùn)行時(shí)動(dòng)態(tài)生成class實(shí)體,通過(guò)Constructor創(chuàng)建類實(shí)例、Field訪問(wèn)成員屬性、Method調(diào)用方法
Class類
在Java中,每個(gè) class 都有一個(gè)相應(yīng)的 Class 對(duì)象。在編譯完成后,在生成的.class文件中,會(huì)產(chǎn)生一個(gè)Class對(duì)象,用于表示這個(gè)類的類型信息。
特點(diǎn):
1、由系統(tǒng)創(chuàng)建Class類
2、同一個(gè)類的實(shí)例指向同一個(gè)Class對(duì)象
3、通過(guò)Class可以完整的得到一個(gè)類所有被加載的結(jié)構(gòu)
4、Class類是Java反射的基礎(chǔ)
獲得Class類的方法:
public class MyReflect { public static void main(String[] args) throws ClassNotFoundException { Teacher t = new Teacher(); // 1、通過(guò)對(duì)象獲得 Class c1 = t.getClass(); // 2、forName + 全限定名 獲得 Class c2 = Class.forName("reflect.Teacher"); // 3、通過(guò)類名獲得 Class c3 = Teacher.class; // 4、基本數(shù)據(jù)類型的Type屬性獲得 Class c4 = Double.TYPE; // 5、通過(guò)子類的Class對(duì)象獲得 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)限檢測(cè),用以訪問(wè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)造方法。通過(guò)Class對(duì)象的 getConstructors() 方法可以獲得當(dāng)前運(yùn)行時(shí)類的構(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對(duì)象的 getField() 或 getFields()等方法可以獲得當(dāng)前運(yùn)行時(shí)類的成員變量
public class MyField { public static void main(String[] args) throws Exception { Class<?> c = Class.forName("reflect.Person"); // 在Person類中,無(wú)參的構(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 成員可以被訪問(wèn) 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對(duì)象的 getMethod() 方法或 getMethods() 方法可以獲得當(dāng)前運(yùn)行時(shí)類的指定方法或所有方法
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:主要簡(jiǎn)單說(shuō)明反射的使用,關(guān)于類加載機(jī)制,性能分析等沒(méi)有說(shuō)明
感謝你能夠認(rèn)真閱讀完這篇文章,希望小編分享Java基于反射機(jī)制獲取不同類的方法內(nèi)容對(duì)大家有幫助,同時(shí)也希望大家多多支持億速云,關(guān)注億速云行業(yè)資訊頻道,遇到問(wèn)題就找億速云,詳細(xì)的解決方法等著你來(lái)學(xué)習(xí)!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。