本篇博客将带大家实现View的事件的注入。

1、目标效果

上篇博客,我们的事件的代码是这么写的:

1. package com.zhy.zhy_xutils_test;  
2.   
3. import .Activity;  
4. import android.os.Bundle;  
5. import android.view.View;  
6. import android.view.View.OnClickListener;  
7. import android.widget.Button;  
8. import android.widget.Toast;  
9.   
10. import com.zhy.ioc.view.ViewInjectUtils;  
11. import com.zhy.ioc.view.annotation.ContentView;  
12. import com.zhy.ioc.view.annotation.ViewInject;  
13.   
14. @ContentView(value = R.layout.activity_main)  
15. public class MainActivity extends Activity implements OnClickListener  
16. {  
17. @ViewInject(.id_btn)  
18. private Button mBtn1;  
19. @ViewInject(.id_btn02)  
20. private Button mBtn2;  
21.   
22. @Override  
23. protected void onCreate(Bundle savedInstanceState)  
24.     {  
25. super.onCreate(savedInstanceState);  
26.           
27. this);  
28.   
29. this);  
30. this);  
31.     }  
32.   
33. @Override  
34. public void onClick(View v)  
35.     {  
36. switch (v.getId())  
37.         {  
38. case .id_btn:  
39. this, "Why do you click me ?",  
40.                     Toast.LENGTH_SHORT).show();  
41. break;  
42.   
43. case .id_btn02:  
44. this, "I am sleeping !!!",  
45.                     Toast.LENGTH_SHORT).show();  
46. break;  
47.         }  
48.     }  
49.   
50. }



光有View的注入能行么,我们写View的目的,很多是用来交互的,得可以点击神马的吧。摒弃传统的神马,setOnClickListener,然后实现匿名类或者别的方式神马的,我们改变为:

1. package com.zhy.zhy_xutils_test;  
2.   
3. import android.view.View;  
4. import android.widget.Button;  
5. import android.widget.Toast;  
6.   
7. import com.zhy.ioc.view.annotation.ContentView;  
8. import com.zhy.ioc.view.annotation.OnClick;  
9. import com.zhy.ioc.view.annotation.ViewInject;  
10.   
11. @ContentView(value = R.layout.activity_main)  
12. public class MainActivity extends BaseActivity  
13. {  
14. @ViewInject(.id_btn)  
15. private Button mBtn1;  
16. @ViewInject(.id_btn02)  
17. private Button mBtn2;  
18.   
19. @OnClick({ .id_btn, .id_btn02 })  
20. public void clickBtnInvoked(View view)  
21.     {  
22. switch (view.getId())  
23.         {  
24. case .id_btn:  
25. this, "Inject Btn01 !", Toast.LENGTH_SHORT).show();  
26. break;  
27. case .id_btn02:  
28. this, "Inject Btn02 !", Toast.LENGTH_SHORT).show();  
29. break;  
30.         }  
31.     }  
32.   
33. }



直接通过在Activity中的任何一个方法上,添加注解,完成1个或多个控件的事件的注入。这里我把onCreate搬到了BaseActivity中,里面调用了ViewInjectUtils.inject(this);

2、实现

1、注解文件

1. package com.zhy.ioc.view.annotation;  
2.   
3. import java.lang.annotation.ElementType;  
4. import java.lang.annotation.Retention;  
5. import java.lang.annotation.RetentionPolicy;  
6. import java.lang.annotation.Target;  
7.   
8. @Target(ElementType.ANNOTATION_TYPE)  
9. @Retention(RetentionPolicy.RUNTIME)  
10. public @interface EventBase  
11. {  
12.     Class<?> listenerType();  
13.   
14.     String listenerSetter();  
15.   
16.     String methodName();  
17. }

1. package com.zhy.ioc.view.annotation;  
2.   
3. import java.lang.annotation.ElementType;  
4. import java.lang.annotation.Retention;  
5. import java.lang.annotation.RetentionPolicy;  
6. import java.lang.annotation.Target;  
7.   
8. import android.view.View;  
9.   
10. @Target(ElementType.METHOD)  
11. @Retention(RetentionPolicy.RUNTIME)  
12. @EventBase(listenerType = View.OnClickListener.class, listenerSetter = "setOnClickListener", methodName = "onClick")  
13. public @interface OnClick  
14. {  
15. int[] value();  
16. }



EventBase主要用于给OnClick这类注解上添加注解,毕竟事件很多,并且设置监听器的名称,监听器的类型,调用的方法名都是固定的,对应上面代码的:


listenerType = View.OnClickListener.class, listenerSetter = "setOnClickListener", methodName = "onClick"

Onclick是用于写在Activity的某个方法上的:

1. @OnClick({ .id_btn, .id_btn02 })  
2. public void clickBtnInvoked(View view)


如果你还记得,上篇博客我们的ViewInjectUtils.inject(this);里面已经有了两个方法,本篇多了一个:


1. public static void inject(Activity activity)  
2.     {  
3.         injectContentView(activity);  
4.         injectViews(activity);  
5.         injectEvents(activity);  
6.     }



2、injectEvents

1. /**
2.      * 注入所有的事件
3.      * 
4.      * @param activity
5.      */  
6. private static void injectEvents(Activity activity)  
7.     {  
8.           
9. extends Activity> clazz = activity.getClass();  
10.         Method[] methods = clazz.getMethods();  
11. //遍历所有的方法  
12. for (Method method : methods)  
13.         {  
14.             Annotation[] annotations = method.getAnnotations();  
15. //拿到方法上的所有的注解  
16. for (Annotation annotation : annotations)  
17.             {  
18. extends Annotation> annotationType = annotation  
19.                         .annotationType();  
20. //拿到注解上的注解  
21.                 EventBase eventBaseAnnotation = annotationType  
22. class);  
23. //如果设置为EventBase  
24. if (eventBaseAnnotation != null)  
25.                 {  
26. //取出设置监听器的名称,监听器的类型,调用的方法名  
27.                     String listenerSetter = eventBaseAnnotation  
28.                             .listenerSetter();  
29.                     Class<?> listenerType = eventBaseAnnotation.listenerType();  
30.                     String methodName = eventBaseAnnotation.methodName();  
31.   
32. try  
33.                     {  
34. //拿到Onclick注解中的value方法  
35.                         Method aMethod = annotationType  
36. "value");  
37. //取出所有的viewId  
38. int[] viewIds = (int[]) aMethod  
39. null);  
40. //通过InvocationHandler设置代理  
41. new DynamicHandler(activity);  
42.                         handler.addMethod(methodName, method);  
43.                         Object listener = Proxy.newProxyInstance(  
44.                                 listenerType.getClassLoader(),  
45. new Class<?>[] { listenerType }, handler);  
46. //遍历所有的View,设置事件  
47. for (int viewId : viewIds)  
48.                         {  
49.                             View view = activity.findViewById(viewId);  
50.                             Method setEventListenerMethod = view.getClass()  
51.                                     .getMethod(listenerSetter, listenerType);  
52.                             setEventListenerMethod.invoke(view, listener);  
53.                         }  
54.   
55. catch (Exception e)  
56.                     {  
57.                         e.printStackTrace();  
58.                     }  
59.                 }  
60.   
61.             }  
62.         }  
63.   
64.     }



嗯,注释尽可能的详细了,主要就是遍历所有的方法,拿到该方法省的OnClick注解,然后再拿到该注解上的EventBase注解,得到事件监听的需要调用的方法名,类型,和需要调用的方法的名称;通过Proxy和InvocationHandler得到监听器的代理对象,显示设置了方法,最后通过反射设置监听器。

这里有个难点,就是关于DynamicHandler和Proxy的出现,如果不理解没事,后面会详细讲解。

3、DynamicHandler

这里用到了一个类DynamicHandler,就是InvocationHandler的实现类:


1. package com.zhy.ioc.view;  
2.   
3. import java.lang.ref.WeakReference;  
4. import java.lang.reflect.InvocationHandler;  
5. import java.lang.reflect.Method;  
6. import java.util.HashMap;  
7.   
8. public class DynamicHandler implements InvocationHandler  
9. {  
10. private WeakReference<Object> handlerRef;  
11. private final HashMap<String, Method> methodMap = new HashMap<String, Method>(  
12. 1);  
13.   
14. public DynamicHandler(Object handler)  
15.     {  
16. this.handlerRef = new WeakReference<Object>(handler);  
17.     }  
18.   
19. public void addMethod(String name, Method method)  
20.     {  
21.         methodMap.put(name, method);  
22.     }  
23.   
24. public Object getHandler()  
25.     {  
26. return handlerRef.get();  
27.     }  
28.   
29. public void setHandler(Object handler)  
30.     {  
31. this.handlerRef = new WeakReference<Object>(handler);  
32.     }  
33.   
34. @Override  
35. public Object invoke(Object proxy, Method method, Object[] args)  
36. throws Throwable  
37.     {  
38.         Object handler = handlerRef.get();  
39. if (handler != null)  
40.         {  
41.             String methodName = method.getName();  
42.             method = methodMap.get(methodName);  
43. if (method != null)  
44.             {  
45. return method.invoke(handler, args);  
46.             }  
47.         }  
48. return null;  
49.     }  
50. }


好了,代码就这么多,这样我们就实现了,我们事件的注入~~

效果图:


ionic框架目前支持android几和ios几 安卓ioc框架_android

效果图其实没撒好贴的,都一样~~~

3、关于代理

那么,本文结束了么,没有~~~关于以下几行代码,相信大家肯定有困惑,这几行干了什么?

1. //通过InvocationHandler设置代理  
2. new DynamicHandler(activity);  
3.                         handler.addMethod(methodName, method);  
4.                         Object listener = Proxy.newProxyInstance(  
5.                                 listenerType.getClassLoader(),  
6. new Class<?>[] { listenerType }, handler);




InvocationHandler和Proxy成对出现,相信大家如果对Java比较熟悉,肯定会想到Java的动态代理~~~

关于InvocationHandler和Proxy的文章,大家可以参考:http://www.ibm.com/developerworks/cn/java/j-lo-proxy1/ ps:IBM的技术文章还是相当不错的,毕竟有人审核还有奖金~

但是我们的实现有一定的区别,我为什么说大家疑惑呢,比如反射实现:

mBtn2.setOnClickListener(this);这样的代码,难点在哪呢?

1、mBtn2的获取?so easy 

2、调用setOnClickListener ? so easy 

but , 这个 this,这个this是OnClickListener的实现类的实例,OnClickListener是个接口~~你的实现类怎么整,听说过反射newInstance对象的,但是你现在是接口!

是吧~现在应该明白上述几行代码做了什么了?实现了接口的一个代理对象,然后在代理类的invoke中,对接口的调用方法进行处理。

4、代码是最好的老师

光说谁都理解不了,你在这xx什么呢??下面看代码,我们模拟实现这样一个情景:

Main类中实现一个Button,Button有两个方法,一个setOnClickListener和onClick,当调用Button的onClick时,触发的事件是Main类中的click方法

涉及到4个类:

Button

1. package com.zhy.invocationhandler;  
2.   
3. public class Button  
4. {  
5. private OnClickListener listener;  
6.   
7. public void setOnClickLisntener(OnClickListener listener)  
8.     {  
9.   
10. this.listener = listener;  
11.     }  
12.   
13. public void click()  
14.     {  
15. if (listener != null)  
16.         {  
17.             listener.onClick();  
18.         }  
19.     }  
20. }



OnClickListener接口

1. package com.zhy.invocationhandler;  
2.   
3. public interface OnClickListener  
4. {  
5. void onClick();  
6. }


OnClickListenerHandler , InvocationHandler的实现类

1. package com.zhy.invocationhandler;  
2.   
3. import java.lang.reflect.InvocationHandler;  
4. import java.lang.reflect.Method;  
5. import java.util.HashMap;  
6. import java.util.Map;  
7.   
8. public class OnClickListenerHandler implements InvocationHandler  
9. {  
10. private Object targetObject;  
11.   
12. public OnClickListenerHandler(Object object)  
13.     {  
14. this.targetObject = object;  
15.     }  
16.   
17. private Map<String, Method> methods = new HashMap<String, Method>();  
18.   
19. public void addMethod(String methodName, Method method)  
20.     {  
21.         methods.put(methodName, method);  
22.     }  
23.   
24. @Override  
25. public Object invoke(Object proxy, Method method, Object[] args)  
26. throws Throwable  
27.     {  
28.   
29.         String methodName = method.getName();  
30.         Method realMethod = methods.get(methodName);  
31. return realMethod.invoke(targetObject, args);  
32.     }  
33.   
34. }



我们的Main

1. package com.zhy.invocationhandler;  
2.   
3. import java.lang.reflect.InvocationTargetException;  
4. import java.lang.reflect.Method;  
5. import java.lang.reflect.Proxy;  
6.   
7. public class Main  
8. {  
9. private Button button = new Button();  
10.       
11. public Main() throws SecurityException, IllegalArgumentException, NoSuchMethodException, IllegalAccessException, InvocationTargetException  
12.     {  
13.         init();  
14.     }  
15.   
16. public void click()  
17.     {  
18. "Button clicked!");  
19.     }  
20.   
21. public void init() throws SecurityException,  
22.             NoSuchMethodException, IllegalArgumentException,  
23.             IllegalAccessException, InvocationTargetException  
24.     {  
25. new OnClickListenerHandler(this);  
26. class.getMethod("click", null);  
27. "onClick", method);  
28.         Object clickProxy = Proxy.newProxyInstance(  
29. class.getClassLoader(),  
30. new Class<?>[] { OnClickListener.class }, h);  
31. "setOnClickLisntener",  
32. class);  
33.         clickMethod.invoke(button, clickProxy);  
34.           
35.     }  
36.   
37. public static void main(String[] args) throws SecurityException,  
38.             IllegalArgumentException, NoSuchMethodException,  
39.             IllegalAccessException, InvocationTargetException  
40.     {  
41.   
42. new Main();  
43.           
44.         main.button.click();  
45.     }  
46.   
47. }



我们模拟按钮点击:调用main.button.click(),实际执行的却是Main的click方法。

看init中,我们首先初始化了一个OnClickListenerHandler,把Main的当前实例传入,然后拿到Main的click方法,添加到OnClickListenerHandler中的Map中。

然后通过Proxy.newProxyInstance拿到OnClickListener这个接口的一个代理,这样执行这个接口的所有的方法,都会去调用OnClickListenerHandler的invoke方法。

但是呢?OnClickListener毕竟是个接口,也没有方法体~~那咋办呢?这时候就到我们OnClickListenerHandler中的Map中大展伸手了:

@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable
{String methodName = method.getName();
Method realMethod = methods.get(methodName);
return realMethod.invoke(targetObject, args);
}

我们显示的把要执行的方法,通过键值对存到Map里面了,等调用到invoke的时候,其实是通过传入的方法名,得到Map中存储的方法,然后调用我们预设的方法~。

这样,大家应该明白了,其实就是通过Proxy得到接口的一个代理,然后在InvocationHandler中使用一个Map预先设置方法,从而实现Button的onClick,和Main的click关联上。

现在看我们InjectEvents中的代码:

1. //通过InvocationHandler设置代理  
2. new DynamicHandler(activity);  
3. //往map添加方法  
4.                         handler.addMethod(methodName, method);  
5.                         Object listener = Proxy.newProxyInstance(  
6.                                 listenerType.getClassLoader(),  
7. new Class<?>[] { listenerType }, handler);



是不是和我们init中的类似~~

好了,关于如何把接口的回调和我们Activity里面的方法关联上我们也解释完了~~~

注:部分代码参考了xUtils这个框架,毕竟想很完善的实现一个完整的注入不是一两篇博客就可以搞定,但是核心和骨架已经实现了~~大家有兴趣的可以继续去完善~