Java reflect 反射机制与泛型擦除

来源:互联网 发布:linux系统ping不通 编辑:程序博客网 时间:2024/05/22 11:49

反射是什么

主要是指程序可以访问,检测和修改它本身状态或行为的一种能力,并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义。
反射是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();}

这样就可以了。

原创粉丝点击