反射機制是在運行狀態(tài)中舒裤,對于任意一個類喳资,都能夠知道這個類的所有屬性和方法;對于任意一個對象腾供,都能夠調用它的任意一個方法和屬性仆邓;這種動態(tài)獲取的信息以及動態(tài)調用對象的方法的功能稱為java語言的反射機制“楸睿【翻譯于 官方文檔】
本篇將從以下幾個方面講述反射的知識:
calss的使用
方法的反射
構造函數(shù)的反射
成員變量的反射
一节值、什么是class類
在面向對象的世界里,萬物皆對象榜聂。類是對象搞疗,類是java.lang.Class類的實例對象。另外class類只有java虛擬機才能new出來须肆。任何一個類都是Class 類的實例對象匿乃。這實例對象有三種表達方式:
```
public class User{
}
public class ClassTest{
User u=new User();
//方式1:
Class c1=User.class;
//方式2:
Class c2=u.getClass();
//方式3:
Class c3=Class.forName("com.forezp.User");
//可以通過類的類型創(chuàng)建該類的實例對象
User user=(User)c1.newInstance();
}
```
二、class類的動態(tài)加載
Class.forName(類的全稱);該方法不僅表示了類的類型豌汇,還代表了動態(tài)加載類扳埂。編譯時刻加載類是靜態(tài)加載、運行時刻加載類是動態(tài)加載類瘤礁。
三阳懂、獲取方法信息
基本的數(shù)據(jù)類型,void關鍵字都Class 類的實例;可以通過
```
getName();getSimpleName()獲取類的名稱柜思。
Class c1=String.class;
Class c2=int.class;
Class c3=void.class;
System.out.println(c1.getName());
System.out.println(c2.getSimpleName());
獲取類的所有方法岩调,并打印出來:
public static void printClassInfo(Object object){
? ? ? ? Class c=object.getClass();
? ? ? ? System.out.println("類的名稱:"+c.getName());
? ? ? ? /**
? ? ? ? * 一個成員方法就是一個method對象
? ? ? ? * getMethod()所有的 public方法,包括父類繼承的 public
? ? ? ? * getDeclaredMethods()獲取該類所有的方法赡盘,包括private ,但不包括繼承的方法号枕。
? ? ? ? */
? ? ? ? Method[] methods=c.getMethods();//獲取方法
? ? ? ? //獲取所以的方法,包括private ,c.getDeclaredMethods();
? ? ? ? for(int i=0;i<methods.length;i++){
? ? ? ? ? ? //得到方法的返回類型
? ? ? ? ? ? Class returnType=methods[i].getReturnType();
? ? ? ? ? ? System.out.print(returnType.getName());
? ? ? ? ? ? //得到方法名:
? ? ? ? ? ? System.out.print(methods[i].getName()+"(");
? ? ? ? ? ? Class[] parameterTypes=methods[i].getParameterTypes();
? ? ? ? ? ? for(Class class1:parameterTypes){
? ? ? ? ? ? ? ? System.out.print(class1.getName()+",");
? ? ? ? ? ? }
? ? ? ? ? ? System.out.println(")");
? ? ? ? }
? ? }
public class ReflectTest {
? ? ? ? public static void main(String[] args){
? ? ? ? ? ? ? ? String s="ss";
? ? ? ? ? ? ? ? ClassUtil.printClassInfo(s);
? ? ? ? }
}
```
運行:
類的名稱:
java.lang.String
booleanequals(java.lang.Object,)
java.lang.StringtoString()
inthashCode()
…
```
四陨享、獲取成員變量的信息
也可以獲取類的成員變量信息
```
public static void printFiledInfo(Object o){
? ? ? ? Class c=o.getClass();
? ? ? ? /**
? ? ? ? * getFileds()獲取public
? ? ? ? * getDeclaredFields()獲取所有
? ? ? ? */
? ? ? ? Field[] fileds=c.getDeclaredFields();
? ? ? ? for(Field f:fileds){
? ? ? ? ? ? //獲取成員變量的類型
? ? ? ? ? ? Class filedType=f.getType();
? ? ? ? ? ? System.out.println(filedType.getName()+" "+f.getName());
? ? ? ? }
? ? }
public static void main(String[] args){
? ? ? ? ? ? ? ? String s="ss";
? ? ? ? ? ? ? ? //ClassUtil.printClassInfo(s);
? ? ? ? ? ? ? ? ClassUtil.printFiledInfo(s);
? ? ? ? }
```
運行:
```
[C value
int hash
long serialVersionUID
[Ljava.io.ObjectStreamField; serialPersistentFields
java.util.Comparator CASE_INSENSITIVE_ORDER
int HASHING_SEED
int hash32
```
五葱淳、獲取構造函數(shù)的信息
```
public static void printConstructInfo(Object o){
? ? ? ? Class c=o.getClass();
? ? ? ? Constructor[] constructors=c.getDeclaredConstructors();
? ? ? ? for (Constructor con:constructors){
? ? ? ? ? ? System.out.print(con.getName()+"(");
? ? ? ? ? ? Class[] typeParas=con.getParameterTypes();
? ? ? ? ? ? for (Class class1:typeParas){
? ? ? ? ? ? ? ? System.out.print(class1.getName()+" ,");
? ? ? ? ? ? }
? ? ? ? ? ? System.out.println(")");
? ? ? ? }
? ? }
public static void main(String[] args){
? ? ? ? ? ? ? ? String s="ss";
? ? ? ? ? ? ? ? //ClassUtil.printClassInfo(s);
? ? ? ? ? ? ? ? //ClassUtil.printFiledInfo(s);
? ? ? ? ? ? ? ? ClassUtil.printConstructInfo(s);
? ? ? ? }
```
運行:
```
java.lang.String([B ,)
java.lang.String([B ,int ,int ,)
java.lang.String([B ,java.nio.charset.Charset ,)
java.lang.String([B ,java.lang.String ,)
java.lang.String([B ,int ,int ,java.nio.charset.Charset ,)
java.lang.String(int ,int ,[C ,)
java.lang.String([C ,boolean ,)
java.lang.String(java.lang.StringBuilder ,)
java.lang.String(java.lang.StringBuffer ,)
…
```
六、方法反射的操作
獲取一個方法:需要獲取方法的名稱和方法的參數(shù)才能決定一個方法抛姑。
方法的反射操作:
method.invoke(對象赞厕,參數(shù)列表);
舉個例子:
```
class A{
? ? public void add(int a,int b){
? ? ? ? System.out.print(a+b);
? ? }
? ? public void toUpper(String a){
? ? ? ? System.out.print(a.toUpperCase());
? ? }
}
public static void main(String[] args) {
? ? ? ? A a=new A();
? ? ? ? Class c=a.getClass();
? ? ? ? try {
? ? ? ? ? ? Method method=c.getMethod("add",new Class[]{int.class,int.class});
? ? ? ? ? ? //也可以 Method method=c.getMethod("add",int.class,int.class);
? ? ? ? ? ? //方法的反射操作
? ? ? ? ? ? method.invoke(a,10,10);
? ? ? ? }catch (Exception e){
? ? ? ? ? ? e.printStackTrace();
? ? ? ? }
? ? }
```
運行:
```
20
```
本篇文章已經講解了java反射的基本用法, 它可以在運行時判斷任意一個對象所屬的類定硝;在運行時構造任意一個類的對象皿桑;在運行時判斷任意一個類所具有的成員變量和方法;在運行時調用任意一個對象的方法;生成動態(tài)代理诲侮。