詳解JAVA 反射機(jī)制
什么是反射?
反射機(jī)制是在程序運(yùn)行狀態(tài)中,對(duì)于任意一個(gè)類,都能夠獲取這個(gè)類的所有屬性和方法;
對(duì)于任意一個(gè)對(duì)象,都能夠調(diào)用它的任意一個(gè)方法和屬性;
這種動(dòng)態(tài)獲取信息以及動(dòng)態(tài)調(diào)用對(duì)象的方法的功能稱為java語(yǔ)言的反射機(jī)制。
反射的作用
1.可以實(shí)現(xiàn)簡(jiǎn)單的反編譯,獲取類中的屬性和方法等基本信息,.class—>java
2.通過(guò)反射機(jī)制獲取類的屬性、方法等
在使用eclipse時(shí),通過(guò)對(duì)象引用.的方式,eclipse就會(huì)將這個(gè)對(duì)象中的所有屬性和方法展示出來(lái),這個(gè)就是利用的反射機(jī)制。其實(shí)反射應(yīng)用最多的地方就是將來(lái)要學(xué)習(xí)的框架,例如spring、spring MVC、Mybatis、hibernate、Struts等等
反射的優(yōu)點(diǎn)
提高程序的靈活性和擴(kuò)展性。使用反射機(jī)制后,程序在創(chuàng)建對(duì)象的時(shí)候不用使用new關(guān)鍵字寫死在程序中了,而是從配置文件中進(jìn)行讀取,這樣可以提高程序的靈活性
反射的缺點(diǎn)
影響性能。使用反射時(shí),代碼量稍微多點(diǎn),并且是要?jiǎng)討B(tài)的將讀取類,所以會(huì)影響性能。下面例子可能不太恰當(dāng),不過(guò)希望能夠幫助你更好地了解,比如你要做汽車:
正常的方式是司機(jī)停車,你開門上車。
動(dòng)態(tài)的方式就是司機(jī)不停車,只降低車速,然后你跑著開門上車。
破壞封裝性。
Class對(duì)象
在類加載器將.class文件讀取到內(nèi)存中的時(shí)候,jvm會(huì)創(chuàng)建這個(gè).class文件的對(duì)象,并且只創(chuàng)建一個(gè)存放到j(luò)vm的方法區(qū)內(nèi)存中,在java.lang包下有個(gè)Class類,這個(gè)類就是.class文件的對(duì)象類型,任何類在被使用時(shí),都會(huì)創(chuàng)建這個(gè)類的Class對(duì)象。除此之外,在java.lang.reflect包下面的類也跟反射有關(guān)。
創(chuàng)建一個(gè)Person類:
package com.sutaoyu.reflect; public class Person { private String name; public Person() { System.out.println("Person類的構(gòu)造方法"); } public Person(String name) { this.name = name; } public void sing() { System.out.println("唱歌"); } public void setName(String name) { this.name = name; } public String getName() { return name; } }
獲取這個(gè)Person類的Class對(duì)象有三種方式:
//第一種方式: //c1引用的對(duì)象代表整個(gè)Person類 Class c1 = Class.forName("com.monkey1024.reflect.Person"); //第二種方式: //java中每個(gè)類型都有 class 屬性. Class c2 = Person.class; //第三種方式: //java語(yǔ)言中任何一個(gè)java對(duì)象都有g(shù)etClass 方法 Person p = new Person(); Class c3 = e.getClass(); //因?yàn)镻erson這個(gè)類在JVM中只有一個(gè),所以c1,c2,c3的內(nèi)存地址是相同的,指向堆中唯一的Class對(duì)象. System.out.println(c1==c2); //true System.out.println(c2==c3); //true
使用反射將.class文件讀取到內(nèi)存中
將上面的Person.java文件刪除,留下Person.class文件。
使用反射將Person.class文件讀取到內(nèi)存中
package com.sutaoyu.reflect; public class RefectTest02 { public static void main(String[] args) { try { //讀取時(shí)需要加上類的包名 Class clazz = Class.forName("com.sutaoyu.reflect.Person"); Object o = clazz.newInstance(); System.out.println(o); }catch(ClassNotFoundException e) { e.printStackTrace(); }catch(InstantiationException e) { e.printStackTrace(); }catch(IllegalAccessException e) { e.printStackTrace(); } } }
使用反射獲取類中的屬性
下面程序在獲取類中的屬性之后,將.class文件中的屬性反編譯并打印出來(lái)了。
package com.sutaoyu.reflect; import java.lang.reflect.Field; import java.lang.reflect.Modifier; public class ReflectTest04 { public static void main(String[] args) throws ClassNotFoundException { Class c = Class.forName("java.lang.Integer"); //獲取類中所有的屬性 Field[] field = c.getDeclaredFields(); //使用反射反編譯 StringBuilder sb = new StringBuilder(200); sb.append(Modifier.toString(c.getModifiers()) + " class "+c.getSimpleName() + "{\n"); //訪問(wèn)權(quán)限修飾符 String s = Modifier.toString(f.getModifiers()); if(!"".equals(s)) { sb.append(Modifier.toString(f.getModifiers()) + ""); } //屬性的類型名稱 Class type = f.getType(); sb.append(f.getName() + ";\n"); //屬性的名字 sb.append("}"); System.out.println(sb.toString()); } }
使用反射獲取類中指定的屬性并賦值
package com.sutaoyu.reflect; import java.lang.reflect.Field; /** * 使用反射獲取類中指定的屬性,并且為其賦值 * */ public class ReflectTest05 { public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, SecurityException, InstantiationException, IllegalAccessException { Class c = Class.forName("com.monkey1024.reflect.User"); /*//獲取指定屬性的Field對(duì)象 Field f = c.getDeclaredField("name"); //創(chuàng)建對(duì)象 Object o = c.newInstance(); //給o對(duì)象上的name屬性賦值為張三 f.set(o, "張三"); //獲取o對(duì)象上的name屬性的值 System.out.println(f.get(o));*/ Field f = c.getDeclaredField("age"); Object o = c.newInstance(); //從外部打破封裝性 f.setAccessible(true); f.set(o, 20); System.out.println(f.get(o)); } }
使用反射獲取類中的方法
下面程序?qū)?class文件中的方法反編譯并打印出來(lái)了
User類:
package com.sutaoyu.reflect; import java.util.Date; public class User { private int age; public String name; protected Date birthday; boolean sex; public void m1(){ } public static void m2(){ } private String m3(){ return ""; } public void m4(int i, String s){ } public String m5(String s, int i){ return s + " , " + i; } }
測(cè)試類:
package com.sutaoyu.reflect; import java.lang.reflect.Method; import java.lang.reflect.Modifier; /** * 使用反射獲取類中的方法 * */ public class ReflectTest06 { public static void main(String[] args) throws ClassNotFoundException { //Class c = Class.forName("com.monkey1024.reflect.User"); Class c = Class.forName("java.lang.Object"); //獲取類中所有方法 Method[] method = c.getDeclaredMethods(); /*for(Method m : method){ //方法修飾符 System.out.println(Modifier.toString(m.getModifiers())); //方法的返回值類型 Class type = m.getReturnType(); System.out.println(type.getSimpleName()); //方法名 System.out.println(m.getName()); //方法參數(shù) Class[] param = m.getParameterTypes(); for(Class p : param){ System.out.println(p.getSimpleName()); } }*/ //反編譯將User類中的方法打印 StringBuilder sb = new StringBuilder(200); sb.append(Modifier.toString(c.getModifiers()) + " class " + c.getSimpleName() + "{\n"); for(Method m : method){ sb.append("\t"); //方法的修飾符 sb.append(Modifier.toString(m.getModifiers()) + " "); //方法的返回值類型 Class type = m.getReturnType(); sb.append(type.getSimpleName() + " "); //方法名 sb.append(m.getName() + " "); //方法參數(shù) sb.append("("); Class[] param = m.getParameterTypes(); for(int i=0; i<param.length; i++){ if(i == param.length-1){ sb.append(param[i].getSimpleName()); }else{ sb.append(param[i].getSimpleName()); sb.append(" ,"); } } sb.append(")"); sb.append("{}\n"); } sb.append("}"); System.out.println(sb.toString()); } }
使用反射調(diào)用類中的方法
package com.sutaoyu.reflect; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; /** * 使用反射調(diào)用類中的方法 * */ public class ReflectTest07 { public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException { Class c = Class.forName("com.monkey1024.reflect.User"); //獲取方法 Method method = c.getDeclaredMethod("m5", String.class, int.class); //創(chuàng)建對(duì)象 Object o = c.newInstance(); Object result = method.invoke(o, "admin", 10); System.out.println(result); } }
使用反射獲取構(gòu)造方法
package com.sutaoyu.reflect; import java.lang.reflect.Constructor; import java.lang.reflect.Modifier; /** * 使用反射獲取類中的構(gòu)造方法 * */ public class ReflectTest08 { public static void main(String[] args) throws ClassNotFoundException { Class c = Class.forName("java.lang.StringBuffer"); //獲取類中所有的構(gòu)造方法 Constructor[] con = c.getDeclaredConstructors(); for(Constructor co : con){ //獲取修飾符 System.out.println(Modifier.toString(co.getModifiers())); //獲取方法名 System.out.println(co.getName()); //獲取方法參數(shù) Class[] type = co.getParameterTypes(); for(Class t : type){ System.out.println(t.getSimpleName()); } } } }
使用反射獲取父類和父接口
package com.monkey1024.reflect; /** * 使用反射獲取父類和父接口 * */ public class ReflectTest09 { public static void main(String[] args) throws ClassNotFoundException { Class c = Class.forName("java.lang.StringBuffer"); //獲取父類 Class sup = c.getSuperclass(); System.out.println(sup.getName()); //獲取父接口 Class[] inter = c.getInterfaces(); for(Class i : inter){ System.out.println(i.getName()); } } }
以上就是詳解JAVA 反射機(jī)制的詳細(xì)內(nèi)容,更多關(guān)于JAVA 反射機(jī)制的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Java中的InputStreamReader和OutputStreamWriter源碼分析_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
本文通過(guò)示例代碼給大家解析了Java中的InputStreamReader和OutputStreamWriter知識(shí),需要的的朋友參考下吧2017-05-05JAVA CountDownLatch(倒計(jì)時(shí)計(jì)數(shù)器)用法實(shí)例
這篇文章主要介紹了JAVA CountDownLatch(倒計(jì)時(shí)計(jì)數(shù)器)用法實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-10-10使用JPA進(jìn)行CriteriaQuery進(jìn)行查詢的注意事項(xiàng)
這篇文章主要介紹了使用JPA進(jìn)行CriteriaQuery進(jìn)行查詢的注意事項(xiàng),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-12-12Java調(diào)用dll文件的實(shí)現(xiàn)解析
這篇文章主要介紹了Java調(diào)用dll文件的實(shí)現(xiàn)解析,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-02-02