一、class类的使用
1.类也是对象,是java.lang.Class的实例对象
2.任何一个类都是Class类的实例对象,有三种表示方式:(Class的构造方法是私有的,只允许JVM调用,因此无法通过关键词new创建Class的实例对象)• 通过类名.class创建,也即说明任何一个类都有一个静态的成员变量class 如:Class c1 = Foo.class;• 通过类的对象的getClass方法 如:Foo foo1 = new Foo(); 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(); // 返回int c2.getName(); // 返回java.lang.String c2.getSimpleName(); // 返回String Methods[] 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():String 4、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);//true Method m=c2.getMethod("add",Object.class); m.invoke(list2,20);//向list2集合中添加一个int 型的值;绕过编译当然是不能直接foreach list2集合的,会报类型转换错误