这是一个比较特殊的知识点,需要从HotSpot与Dalvik的区别说起。
问题起因
相信大家在继承RecycleView时,对ViewHolder中的itemView都有重写的欲望。这个itemView是final类型的,不容许被修改,极大的限制了开发的自由。
那首先想到的就是利用反射来修改该字段。可是运行会报错。
创建一个通用方法进行反射修改属性值,如下:
public static void modify(Object object, String fieldName, Object newFieldValue) throws Exception {
Field field = object.getClass().getDeclaredField(fieldName);
Field modifiersField = Field.class.getDeclaredField("modifiers");
modifiersField.setAccessible(true); //Field 的 modifiers 是私有的
modifiersField.setInt(field, field.getModifiers() & ~Modifier.FINAL);
if(!field.isAccessible()) {
field.setAccessible(true);
}
field.set(object, newFieldValue);
}
在实际运行时,会报错,提示说没有modifiers字段。这也就表明,Android编译后的dex字节码中,并没有java那一套复杂属性访问控制逻辑。
反射时忽视final修饰符
调整代码,不管final,直接尝试修改itemView
Field field = RecyclerView.ViewHolder.class.getField("itemView");
field.set(viewHolder, someNewItemView);
此时是可以正常运行的。