初级程序员使用别人的框架,中级程序员不仅会使用别人的框架还知道内部的实现原理,高级程序员则按需编写自己的框架。添加该模块的目的就是想提交大家的逼格,让大家养成一个动手编写“自主知识产权”框架的意识。
业界比较出名的基于完全注解方式就可以进行 UI 绑定和事件绑定,无需 findViewById 和 setClickListener 等的IOC(Inverse Of Control 控制反转,就是将 UI 的初始化和事件绑定的“权利”交给框架来完成)框架有:
ButterKnife使用如下:
会出现如下代码:
@BindView(R.id.button01)
Button mButton01;
@BindView(R.id.button02)
Button mButton02;
@BindView(R.id.button03)
Button mButton03;
@butterknife.OnClick({R.id.button01, R.id.button02, R.id.button03})
public void onClick(View view) {
switch (view.getId()) {
case R.id.button01:
Toast.makeText(this, "butterknife-button01", Toast.LENGTH_SHORT).show();
break;
case R.id.button02:
Toast.makeText(this, "butterknife-button02", Toast.LENGTH_SHORT).show();
break;
case R.id.button03:
Toast.makeText(this, "butterknife-button03", Toast.LENGTH_SHORT).show();
break;
}
}
点击每个按钮会弹出响应的Toast,如下:
这个就是ButterKnife的用法。
- 编写自定义注解类 ViewInject 和 Click;
ViewInject 注解类用于添加在 Filed 上。Click 注解类用于添加到 Method 上。
【文件】ViewInject.java
/**
* @Retention 用于声明该注解生效的生命周期,有三个枚举值可以选择<br>
* 1. RetentionPolicy.SOURCE 注释只保留在源码上面,编译成class的时候自动被编译器抹除
* 2. RetentionPolicy.CLASS 注释只保留到字节码上面,VM加载字节码时自动抹除
* 3. RetentionPolicy.RUNTIME 注释永久保留,可以被VM加载时加载到内存中
* 注意:由于我们的目的是想在VM运行时对Filed上的该注解进行反射操作,因此Retention值必须设置为RUNTIME
*
* @Target 用于指定该注解可以声明在哪些成员上面,常见的值有FIELD和Method,
由于我们的当前注解类是想声明在Filed上面
* 因此这里设置为ElementType.FIELD。
* 注意:如果@Target值不设置,则默认可以添加到任何元素上,不推荐这么写。
*
* @interface 是声明注解类的组合关键字。
*/
@Target({java.lang.annotation.ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
public @interface ViewInject {
public abstract int value();
}
【文件】Click.java
@Target({java.lang.annotation.ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface OnClick
{
public abstract int[] value();
}
编写核心方法 ViewUtilsTest.inject(Ativity);
public class ViewUtilsTest {
public static void inject(final Activity activity)
{
/**
* 通过字节码获取activity类中所有的字段,在获取Field的时候一定要使用
* getDeclaredFields(),
* 因为只有该方法才能获取到任何权限修饰的Filed,包括私有的。
*/
Class clazz = activity.getClass();
Field[] declaredFields = clazz.getDeclaredFields();
//一个Activity中可能有多个Field,因此遍历。
for (int i = 0; i < declaredFields.length; i++) {
Field field = declaredFields[i];
//设置为可访问,暴力反射,就算是私有的也能访问到
field.setAccessible(true);
//获取到字段上面的注解对象
ViewInject annotation = (ViewInject)field.getAnnotation(ViewInject.class);
//一定对annotation是否等于null进行判断,因为并不是所有Filed上都有我们想要的注解
if (annotation == null)
{
continue;
}
//获取注解中的值
int id = annotation.value();
//获取控件
View view = activity.findViewById(id);
try
{
//将该控件设置给field对象
field.set(activity, view);
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
}
//获取所有的方法(私有方法也可以获取到)
Method[] declaredMethods = clazz.getDeclaredMethods();
for (int i = 0; i < declaredMethods.length; i++) {
final Method method = declaredMethods[i];
//获取方法上面的注解
OnClick annotation = (OnClick)method.getAnnotation(OnClick.class);
if (annotation == null) {
//如果该方法上没有注解,循环下一个
continue;
}
//获取注解中的数据,因为可以给多个button绑定点击事件,因此定义注解类时使用的是int[]作为数据类型。
int[] value = annotation.value();
for (int j = 0; j < value.length; j++) {
int id = value[j];
final View button = activity.findViewById(id);
button.setOnClickListener(new View.OnClickListener()
{
public void onClick(View v)
{
try
{
//反射调用用户指定的方法
method.invoke(activity,button);
} catch (Exception e) {
e.printStackTrace();
}
}
});
}
}
}
}
咱们自己的注解框架就实现好了,效果如下:
- 欢迎关注微信公众号,长期推荐技术文章和技术视频
微信公众号名称:Android干货程序员