java反射

2016-06-28  本文已影响36人  一条自由自在的鱼

来自慕课网“反射——Java高级开发必须懂的”,感谢cedar老师的讲解。

类类型

我们写的类都是java.lang.Class类的实例对象。举例说明:

public static void main(String[] args){
//Foo的实例对象
Foo foo1 =  new Foo();
//Foo类本身是Class的实例对象,怎么表示?像这样的构造方法Class()在Class类内部声明为私有,只有JVM可以访问创建,所以不能直接声明构造。
//任何类都是Class的实例对象,有三种表示方法。

//第一种表示方式-->每个类都有一个隐含的静态成员变量
Class c1 = Foo.class;

//第二种表示方式-->已知该类的对象,通过getClass()方法获得
Class c2 = foo1.getClass();

//foo1的类型是Foo,Foo的类型是Class(类类型,classtype)
//类也是对象,是Class类的实例对象
//c1, c2都代表Foo类的类类型,一个类只可能是Class类的一个实例对象。
System.out.println(c1==c2);

//第三种表达方式
Class c3 = null;
try{
    c3 = Class.forName("com.imooc.reflect.Foo");
}catch (ClassNotFoundException e){
    e.printStackTrace();
}
System.out.println(c2==c3);

}

//通过该类的类类型创建该类的实例对象
try {
    Foo foo1= (Foo)c1.newInstance();
} catch (InstantiationException e) {
    e.printStackTrace();
} catch (IllegalAccessException e) {
    e.printStackTrace();
}

Class动态加载类

静态加载类和动态加载类:
前者是编译时刻加载所有可能使用到的类,后者是运行时刻加载类。
new创建对象是静态加载类

Class c = Class.forName(args[0]);
OfficeAble oa = (OfficeAble)c.newInstance();//OfficeAble定义的功能类型接口

基本数据类型的类类型

基本数据类型、void关键字,都存在类类型

Class cc1 = int.class;
Class cc2 = Integer.class;
Class cc3 = void.class;
System.out.println(cc1.getName());
System.out.println(cc2.getName());
System.out.println(cc2.getSimpleName()); //不包含包名的类名称

Class类的基本API操作

获取成员方法信息
Class c = obj.getClass();//传递的是哪个子类的对象,c就是该子类的类类型
System.out.println("类的名称是:"+c.getName());
Method[] ms = c.getMethods();//c.getDeclaredMethods();
for(Method m:ms){    //得到方法返回值的类类型,如String.class    
Class returnType = m.getReturnType();    
System.out.print(returnType.getName()+" ");    //得到方法名称    
System.out.print(m.getName()+"(");    //获取参数类型,得到参数列表的类型的类类型,如int.class,int.class    
Class[] paraTypes = m.getParameterTypes();    
for(Class paraType:paraTypes){        
System.out.print(paraType.getName()+",");    
}    
System.out.println(")");}
获取成员变量信息

成员变量也是对象,java.lang.reflect.Field类封装了关于成员变量的操作

Class c = obj.getClass();
Field[] fs = c.getFields();

getFields()获取所有public成员变量信息

获取构造函数信息

构造函数也是变量,java.lang.reflect.Constructor中封装了构造函数的信息

方法的反射

Method m = c.getMethod("print",new Class[]{int.class,int.class});
Object o = m.invoke(a1,10,20);
//和a1.print调用的效果完全相同

通过Class和Method来理解泛型本质

反射的操作都是编译之后的操作
编译之后集合的泛型都是去泛型化的
这说明java中的泛型是为了防止错误输入的,只在编译阶段有效,绕过编译就无效了。

上一篇 下一篇

猜你喜欢

热点阅读