注解框架的实现
附上github地址:https://github.com/yuanmingsun/easyannotion.git
注解是java1.5开始的。在java框架中比如spring,hibernate中都大量用到了注解。android中如retrofit和butterkinfe都用到了注解。比如我们在android开发中大量用到了findviewbyid,我们可以使用注解来生成findviewbyid,简化自己的工作量。
注解的分类
元注解包括@Retention,@Target,@Document,@Inherited
1. @Retention:指该注解保留的时间长短,RetentionPolicy是一个enum类型,取值分别为SOURCE,CLASS,RUNTIME
1).@Retention(RetentionPolicy.SOURCE):注解仅存在于源码中。
2).@Retention(RetentionPolicy.ClASS):默认的保留方式,注解存在于class字节码中,但是运行时无法获取。
3).@Retention(RetentionPolicy.RUNTIME)注解可以在运行时通过反射得到。
2.@Target:指注解可以注解的元素范围.取值可以为TYPE类,METHOD方法,CONSTRUCTOR构造函数,FIELD字段等。如果@Target注解没有定义,那么注解的范围是所有。
3.@Document:指明该注解可以被javadoc此类的工具文档化
4.@Inherited: 指明该注解类型可以被自动继承。
内建注解
1.@override :想要重写父类的方法时,通过这个注解告诉编译器我们要重写这个方法。如果父类中方法变化那么编译器就会告警。
2.@Deprecated:某一个方法不被推荐使用时,可以用这个注解来声明。
3.SupressWarning告诉编译器忽略特定的警告信息。保留时常为soure,会被编译器丢掉.
4.android的内建注解就比较多了。如@StringRes,@Nullable。在谷歌提供的support-annotions库中。
说了这么多,我们开始打造自己的注解框架把。首先我们来打造一个编译时注解框架。
1.首先新建两个注解,一个是bindId.class,用于绑定activity和view,一个是onClick.class用于绑定点击事件
然后我们去通过反射来获取拿到注解,绑定findviewbyid和onclick
public class BindApi {
public static void init(Activity obj) {
bindViewId(obj);
bindOnClick(obj);
}
private static void bindOnClick(final Activity obj) {
Class cls=obj.getClass();
Method [] methods=cls.getMethods();
for(final Method method :methods)
{
if(method.isAnnotationPresent(OnClick.class))
{
OnClick onClick=method.getAnnotation(OnClick.class);
int []ids=onClick.value();
for(int id :ids)
{
Method method2 =null;
try {
method2 = cls.getMethod("findViewById", int.class);
}catch (NoSuchMethodException e) {
e.printStackTrace();
}
method2.setAccessible(true);
View v=null;
try {
v = (View)method2.invoke(obj, id);
}catch (IllegalAccessException e) {
e.printStackTrace();
}catch (InvocationTargetException e) {
e.printStackTrace();
}
v.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
method.setAccessible(true);
try {
method.invoke(obj,v);
}catch (IllegalAccessException e) {
e.printStackTrace();
}catch (InvocationTargetException e) {
e.printStackTrace();
}
}
});
}
}
}
}
private static void bindViewId(Activity obj) {
Class cls = obj.getClass();
if (cls.isAnnotationPresent(BindId.class)) {
//这里使用了反射的方法来调用setContentView
BindId mId = cls.getAnnotation(BindId.class);
int id = mId.value();
try {
Method method = cls.getMethod("setContentView", int.class);
method.setAccessible(true);
method.invoke(obj, id);
}catch (NoSuchMethodException e) {
e.printStackTrace();
}catch (IllegalAccessException e) {
e.printStackTrace();
}catch (InvocationTargetException e) {
e.printStackTrace();
}
//这里直接调用findviewbyid
Field[] fields = cls.getDeclaredFields();
for (Field field : fields) {
if (field.isAnnotationPresent(BindId.class)) {
BindId mId2 = field.getAnnotation(BindId.class);
int id2 = mId2.value();
try {
Method method = cls.getMethod("findViewById", int.class);
method.setAccessible(true);
View view = (View)method.invoke(obj,id2);
field.setAccessible(true);
field.set(obj, view);
}catch (IllegalAccessException e) {
e.printStackTrace();
}catch (NoSuchMethodException e) {
e.printStackTrace();
}catch (InvocationTargetException e) {
e.printStackTrace();
}
}
}
}
}
}
最后,在activity使用