浅析Java反射

来源:互联网 发布:js 正则匹配数字 编辑:程序博客网 时间:2024/06/05 06:45

一、class类的使用

1.类也是对象,是java.lang.Class的实例对象

2.任何一个类都是Class类的实例对象,有三种表示方式:(Class的构造方法是私有的,只允许JVM调用,因此无法通过关键词new创建Class的实例对象)•通过类名.class创建,也即说明任何一个类都有一个静态的成员变量class 如:Class c1 = Foo.class;• 通过类的对象的getClass方法 如:Foo foo1 = newFoo(); Class c2 = foo1.getClass();

• 通过Class的forName方法传入类的全限定名获取 如:Class c3 = null; c3 = Class.forName("com.imooc.reflect.Foo");(会有ClassNotFoundException)

• c1、c2表示的Foo类的类类型(class type) 万事万物皆对象,类也是对象,是Class类的实例对象,这个对象称为类的类类型可以通过类的类类型 创建该类的对象§ 如:Foo foo2 = (Foo)c1.newInstance();

• 一个类只能是Class的一个实例对象,无论通过三种方法中的哪一种获取都一样

• 三种方法中,c1、c2为静态加载类(编译时加载),而c3为动态加载类(运行时加载)
二、动态加载类
1.编译时刻加载类是静态加载类、运行时刻加载类是动态加载类2.new 创建对象是静态加载类,在编译时刻就需要加载所有的可能使用到的类。3.Class c=Class.foName(args[0])为动态加载类,在运行时刻加载4.通过类类型,创建该对象 OfficeAble oa=c.newInstance();//OfficeAble是一个接口
三、获取方法信息
1.void等关键字都存在类类型,比如 Class c=int.class;Class c1=String.class;2.Method类,方法对象,一个成员方法就是一个Method对象,getMethods()方法获取的就是所有的public的方法,包括父类继承而来的,getDeclaredMethods()获取的是所有该类自己声明的方法,不会访问继承而来的方法;3.Method[]ms=c.getMethods();获取c类中所有的方法,Class returnType=ms[i].getReturnType();获取方法的返回值类型的类类型;ms[i].getName();获取方法的名称4.Class[]paramTypes=ms[i].getParameterTypes();获取参数类型--->得到的是参数列表的类型的类类型;
Class c1 = int.class; // int基本类型的类类型Class c2 = String.class; // String类的类类型c1.getName(); // 返回intc2.getName(); // 返回java.lang.Stringc2.getSimpleName(); // 返回StringMethods[] ms = c1.getMethods(); // 获取该类的所有公有方法,包括继承的方法Methods[] ms = c1.getDeclaredMethods(); / 获取所有自己声明的方法,包括私有方法for(int i=0;i<ms.length;i++){  Class returnType = ms[i].getReturnType; // 获取这个方法的返回值的类类型  returnType.getName(); // 该方法的返回值的类类型的名称  ms[i].getName(); // 该方法的名称  Class[] paramTypes = ms[i].getParameterType(); // 获取该方法的参数的类型的类类型的数组  for(Class class1: paramType){    class1.getName(); // 获取该参数的类型的名称}}

四、获取成员变量&构造函数一、成员变量是java.lang.reflect.Field的对象1、Field类封装了关于成员变量的操作2、Field[] fs = c.getFields()方法获取所有public的成员变量Field[]信息3、c.getDeclaredFields获取的是该类自己声明的成员变量信息4、field.getType()获得成员类型的类类型5、field.getName()获得成员的名称二、构造函数是java.lang.Constructor类的对象1、通过Class.getConstructor()获得Constructor[]所有公有构造方法信息2、建议getDeclaredConstructors()获取自己声明的构造方法3、Constructor.getName():String4、Constructor.getParameterTypes():Class[]成员变量也是对象,是java.lang.reflect.Field的对象;
五、方法的反射操作
方法的反射:1.获取A类中的print(int,int)方法: ①要获取一个方法就是获取类的信息,获取类的信息首先要获取类的类类型  A a1=new A();  Class c= a1.getClass(); ②获取方法 由名称和参数列表来决定,getMethod获取的是public方法,getDelcaredMethod获取自己声明的方法  Method m =c.getMethod(methodName,paramtypes);//paramtypes可以用数组的形式  表示new Class[]{int.class,int.class},也可以直接列举类类型  2.方法的反射操作:是用m对象来进行方法调用,和a1.print(10,20)调用的方法相同 m.invoke(a1,new Object[]{10,20})Object o=m.invoke(对象名,参数);//方法如果没有返回值返回null,如果有返回值返回具体值,参数可用数组的方式表示,也可以直接列举,没有参数就不写
public Class A{ public void print(){}; public void Print(Sting a,String b){} public void Print(int a,int b){};} public Class B{ public static void main(String[] args){  A a1 = new A();  Class c= a1.getclass;  Method getMet=c.getMethod("print",String.class,String.class);//忘了加引号  Object obj=getMet.invoke(a1,"df","df");}}

六、通过反射了解集合泛型的本质1:反射的操作都是编译之后的操作;就是运行阶段2:java中集合的泛型是防止错误输入的;只在编译阶段有效,只要绕过编译就无效啦我们可以通过方法的反射来操作,绕过编译
ArrayList list1=new ArrayList();ArrayList<String> list2=new ArrayList<String>();Class c1=list1.getClass();Class c2=list2.getClass();System.out.print(c1==c2);//trueMethod m=c2.getMethod("add",Object.class);m.invoke(list2,20);//向list2集合中添加一个int 型的值;绕过编译
当然是不能直接foreach list2集合的,会报类型转换错误