您的位置:首页 > 编程语言 > Java开发

关于Java的反射机制,你需要理解这些...

2017-04-13 12:59 471 查看
反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。【翻译于 官方文档】

本篇将从以下几个方面讲述反射的知识:

class 的使用

方法的反射

构造函数的反射

成员变量的反射


一、什么是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");  

//可以通过类的类型创建该类的实例对象  
User user=(User)c1.newInstance();  
}


二、class类的动态加载

Class.forName(类的全称);该方法不仅表示了类的类型,还代表了动态加载类。编译时刻加载类是静态加载、运行时刻加载类是动态加载类。


三、获取方法信息

基本的数据类型,void关键字都Class 类的实例;可以通过get

ame();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());  
       }  

   }

               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


五、获取构造函数的信息

    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 ,)

...


六、方法反射的操作

获取一个方法:需要获取方法的名称和方法的参数才能决定一个方法。

方法的反射操作:
method.invoke(对象,参数列表);


举个例子:
class A{  

   public void add(int a,int b){  
       System.out.print(a+b);  
   }  

   public void toUpper(String a){  
       System.out.print(a.toUpperCase());  
   }  
}

class A{  
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反射的基本用法, 它可以在运行时判断任意一个对象所属的类;在运行时构造任意一个类的对象;在运行时判断任意一个类所具有的成员变量和方法;在运行时调用任意一个对象的方法;生成动态代理。

来自:简书 方志朋
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  class