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

Java reflect 反射机制与泛型擦除

2017-12-11 11:07 465 查看

反射是什么

主要是指程序可以访问,检测和修改它本身状态或行为的一种能力,并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义。

反射是java中一种强大的工具,能够使我们很方便的创建灵活的代码,这些代码可以再运行时装配,无需在组件之间进行源代码链接。但是反射使用不当会成本很高!

实现方式

反编译:.class–>.java

通过反射机制访问java对象的属性,方法,构造方法等;

涉及类

java.lang.Class;

java.lang.reflect.Constructor;

java.lang.reflect.Field;

java.lang.reflect.Method;

java.lang.reflect.Modifier;

Field

Java反射中Field用于获取某个类的属性或该属性的属性值

功能实现

反射获取类的三种方法

Class c1 = Class.forName("Book");


Class c2 = Book.class; // Java中每个类型都有class属性


Book book= new Book(); // java语言中任何一个java对象都有getClass 方法

Class c3 = book.getClass(); // c3是运行时类 (book的运行时类是Book)


创建对象

获取类以后我们来创建它的对象,利用newInstance:

Class c =Class.forName("Book");

// 创建此Class 对象所表示的类的一个新实例
Object o = c.newInstance(); // 调用Book的无参数构造方法.


获取所有属性

Class c = Class.forName("java.lang.Integer"); // 获取整个类
Field[] fs = c.getDeclaredFields();  // 获取所有的属性
StringBuffer sb = new StringBuffer();  // 定义可变长的字符串,用来存储属性

sb.append(Modifier.toString(c.getModifiers()) + " class " + c.getSimpleName() +"{\n");  // 通过追加的方法,将public定义里边的每一个属性拼接到此字符串中
for(Field field:fs){
sb.append("\t"); // Tab
sb.append(Modifier.toString(field.getModifiers())+" "); // 获得属性的修饰符,例如public,static等
sb.append(field.getType().getSimpleName() + " ");// 属性的类型的名字
sb.append(field.getName()+";\n"); // 属性名字+回车换行
}
sb.append("}");
System.out.println(sb);


获取指定属性

Class c = Class.forName("User"); // 获取类
Field idF = c.getDeclaredField("id"); // 获取id属性
Object o = c.newInstance(); // 实例化这个类赋给o
idF.setAccessible(true); // 打破封装 :使用反射机制可以打破封装性,导致了对象的属性不安全。
idF.set(o, "110"); // set,给o对象的id属性赋值"110"
System.out.println(idF.get(o)); // get


获取方法和构造方法

方法关键字

含义

getDeclaredMethods()

获取所有的方法

getReturnType()

获得方法的放回类型

getParameterTypes()

获得方法的传入参数类型

getDeclaredMethod(“方法名”,参数类型.class,……)

获得特定的方法

构造方法关键字

含义

getDeclaredConstructors()

获取所有的构造方法

getDeclaredConstructor(参数类型.class,……)

获取特定的构造方法

父类和父接口

含义

getSuperclass()

获取某类的父类

getInterfaces()

获取某类实现的接口

这样我们就可以获得类的各种内容,进行了反编译。对于JAVA这种先编译再运行的语言来说,反射机制可以使代码更加灵活,更加容易实现面向对象。

拓展:使用反射进行泛型擦除:如何把Activity对象放进list集合里去?

当我们正常的将 Activity 对象放进 list中去,IDE会给出错误:

List<String> list = new ArrayList<>();
list.add("Hello");
list.add(this); // List cannot be applied to XXXActivity.


显然,泛型的约束让我们无法将 Activity 对象放进泛型为 String 的集合中去。

我们可以看下编译后的XXXActivity.class,路径如下:

app/build/intermediates/classes

ArrayList list = new ArrayList<>();
list.add("Hello");


可以看到之前 java 代码里对 List 中的元素只能为 String 的泛型代码没了,这就是泛型擦除。那么也就是说泛型的约束是在编译时约束的,真正运行的 class 是没有泛型约束的,那么想解决题目的话,只要在运行时将 Activity 对象加入就好了,那么自然想到反射,我们来尝试下:

List<String> mList = new ArrayList();
mList .add("Hello");

try{
Field field = getClass().getDeclaredField("mList");
field.setAccessible(true);
List list = (List) field.get(this);
list.add(this);
} catch (NoSuchFieldException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}


这样就可以了。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: