詳解JAVA 反射機制
什么是反射?
反射機制是在程序運行狀態(tài)中,對于任意一個類,都能夠獲取這個類的所有屬性和方法;
對于任意一個對象,都能夠調(diào)用它的任意一個方法和屬性;
這種動態(tài)獲取信息以及動態(tài)調(diào)用對象的方法的功能稱為java語言的反射機制。
反射的作用
1.可以實現(xiàn)簡單的反編譯,獲取類中的屬性和方法等基本信息,.class—>java
2.通過反射機制獲取類的屬性、方法等
在使用eclipse時,通過對象引用.的方式,eclipse就會將這個對象中的所有屬性和方法展示出來,這個就是利用的反射機制。其實反射應(yīng)用最多的地方就是將來要學(xué)習(xí)的框架,例如spring、spring MVC、Mybatis、hibernate、Struts等等
反射的優(yōu)點
提高程序的靈活性和擴展性。使用反射機制后,程序在創(chuàng)建對象的時候不用使用new關(guān)鍵字寫死在程序中了,而是從配置文件中進行讀取,這樣可以提高程序的靈活性
反射的缺點
影響性能。使用反射時,代碼量稍微多點,并且是要動態(tài)的將讀取類,所以會影響性能。下面例子可能不太恰當(dāng),不過希望能夠幫助你更好地了解,比如你要做汽車:
正常的方式是司機停車,你開門上車。
動態(tài)的方式就是司機不停車,只降低車速,然后你跑著開門上車。
破壞封裝性。
Class對象
在類加載器將.class文件讀取到內(nèi)存中的時候,jvm會創(chuàng)建這個.class文件的對象,并且只創(chuàng)建一個存放到j(luò)vm的方法區(qū)內(nèi)存中,在java.lang包下有個Class類,這個類就是.class文件的對象類型,任何類在被使用時,都會創(chuàng)建這個類的Class對象。除此之外,在java.lang.reflect包下面的類也跟反射有關(guān)。
創(chuàng)建一個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;
}
}
獲取這個Person類的Class對象有三種方式:
//第一種方式:
//c1引用的對象代表整個Person類
Class c1 = Class.forName("com.monkey1024.reflect.Person");
//第二種方式:
//java中每個類型都有 class 屬性.
Class c2 = Person.class;
//第三種方式:
//java語言中任何一個java對象都有g(shù)etClass 方法
Person p = new Person();
Class c3 = e.getClass();
//因為Person這個類在JVM中只有一個,所以c1,c2,c3的內(nèi)存地址是相同的,指向堆中唯一的Class對象.
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 {
//讀取時需要加上類的包名
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文件中的屬性反編譯并打印出來了。
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");
//訪問權(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對象
Field f = c.getDeclaredField("name");
//創(chuàng)建對象
Object o = c.newInstance();
//給o對象上的name屬性賦值為張三
f.set(o, "張三");
//獲取o對象上的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文件中的方法反編譯并打印出來了
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;
}
}
測試類:
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)建對象
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 反射機制的詳細內(nèi)容,更多關(guān)于JAVA 反射機制的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Java中的InputStreamReader和OutputStreamWriter源碼分析_動力節(jié)點Java學(xué)院整理
本文通過示例代碼給大家解析了Java中的InputStreamReader和OutputStreamWriter知識,需要的的朋友參考下吧2017-05-05
JAVA CountDownLatch(倒計時計數(shù)器)用法實例
這篇文章主要介紹了JAVA CountDownLatch(倒計時計數(shù)器)用法實例,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2019-10-10

