Android 进阶 教你打造 Android 中的 IOC 框架 【ViewInject】 (下)

import com.zhy.ioc.view.annotation.ContentView;

import com.zhy.ioc.view.annotation.ViewInject;

@ContentView(value = R.layout.activity_main)

public class MainActivity extends Activity implements OnClickListener

{

@ViewInject(R.id.id_btn)

private Button mBtn1;

@ViewInject(R.id.id_btn02)

private Button mBtn2;

@Override

protected void onCreate(Bundle savedInstanceState)

{

super.onCreate(savedInstanceState);

ViewInjectUtils.inject(this);

mBtn1.setOnClickListener(this);

mBtn2.setOnClickListener(this);

}

@Override

public void onClick(View v)

{

switch (v.getId())

{

case R.id.id_btn:

Toast.makeText(MainActivity.this, “Why do you click me ?”,

Toast.LENGTH_SHORT).show();

break;

case R.id.id_btn02:

Toast.makeText(MainActivity.this, “I am sleeping !!!”,

Toast.LENGTH_SHORT).show();

break;

}

}

}

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

package com.zhy.zhy_xutils_test;

import android.view.View;

import android.widget.Button;

import android.widget.Toast;

import com.zhy.ioc.view.annotation.ContentView;

import com.zhy.ioc.view.annotation.OnClick;

import com.zhy.ioc.view.annotation.ViewInject;

@ContentView(value = R.layout.activity_main)

public class MainActivity extends BaseActivity

{

@ViewInject(R.id.id_btn)

private Button mBtn1;

@ViewInject(R.id.id_btn02)

private Button mBtn2;

@OnClick({ R.id.id_btn, R.id.id_btn02 })

public void clickBtnInvoked(View view)

{

switch (view.getId())

{

case R.id.id_btn:

Toast.makeText(this, “Inject Btn01 !”, Toast.LENGTH_SHORT).show();

break;

case R.id.id_btn02:

Toast.makeText(this, “Inject Btn02 !”, Toast.LENGTH_SHORT).show();

break;

}

}

}

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

2、实现

====

1、注解文件


package com.zhy.ioc.view.annotation;

import java.lang.annotation.ElementType;

import java.lang.annotation.Retention;

import java.lang.annotation.RetentionPolicy;

import java.lang.annotation.Target;

@Target(ElementType.ANNOTATION_TYPE)

@Retention(RetentionPolicy.RUNTIME)

public @interface EventBase

{

Class<?> listenerType();

String listenerSetter();

String methodName();

}

package com.zhy.ioc.view.annotation;

import java.lang.annotation.ElementType;

import java.lang.annotation.Retention;

import java.lang.annotation.RetentionPolicy;

import java.lang.annotation.Target;

import android.view.View;

@Target(ElementType.METHOD)

@Retention(RetentionPolicy.RUNTIME)

@EventBase(listenerType = View.OnClickListener.class, listenerSetter = “setOnClickListener”, methodName = “onClick”)

public @interface OnClick

{

int[] value();

}

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

listenerType = View.OnClickListener.class, listenerSetter = “setOnClickListener”, methodName = “onClick”

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

@OnClick({ R.id.id_btn, R.id.id_btn02 })

public void clickBtnInvoked(View view)

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

public static void inject(Activity activity)

{

injectContentView(activity);

injectViews(activity);

injectEvents(activity);

}

2、injectEvents


/**

  • 注入所有的事件

  • @param activity

*/

private static void injectEvents(Activity activity)

{

Class<? extends Activity> clazz = activity.getClass();

Method[] methods = clazz.getMethods();

//遍历所有的方法

for (Method method : methods)

{

Annotation[] annotations = method.getAnnotations();

//拿到方法上的所有的注解

for (Annotation annotation : annotations)

{

Class<? extends Annotation> annotationType = annotation

.annotationType();

//拿到注解上的注解

EventBase eventBaseAnnotation = annotationType

.getAnnotation(EventBase.class);

//如果设置为EventBase

if (eventBaseAnnotation != null)

{

//取出设置监听器的名称,监听器的类型,调用的方法名

String listenerSetter = eventBaseAnnotation

.listenerSetter();

Class<?> listenerType = eventBaseAnnotation.listenerType();

String methodName = eventBaseAnnotation.methodName();

try

{

//拿到Onclick注解中的value方法

Method aMethod = annotationType

.getDeclaredMethod(“value”);

//取出所有的viewId

int[] viewIds = (int[]) aMethod

.invoke(annotation, null);

//通过InvocationHandler设置代理

DynamicHandler handler = new DynamicHandler(activity);

handler.addMethod(methodName, method);

Object listener = Proxy.newProxyInstance(

listenerType.getClassLoader(),

new Class<?>[] { listenerType }, handler);

//遍历所有的View,设置事件

for (int viewId : viewIds)

{

View view = activity.findViewById(viewId);

Method setEventListenerMethod = view.getClass()

.getMethod(listenerSetter, listenerType);

setEventListenerMethod.invoke(view, listener);

}

} catch (Exception e)

{

e.printStackTrace();

}

}

}

}

}

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

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

3、DynamicHandler


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

package com.zhy.ioc.view;

import java.lang.ref.WeakReference;

import java.lang.reflect.InvocationHandler;

import java.lang.reflect.Method;

import java.util.HashMap;

public class DynamicHandler implements InvocationHandler

{

private WeakReference handlerRef;

private final HashMap<String, Method> methodMap = new HashMap<String, Method>(

1);

public DynamicHandler(Object handler)

{

this.handlerRef = new WeakReference(handler);

}

public void addMethod(String name, Method method)

{

methodMap.put(name, method);

}

public Object getHandler()

{

return handlerRef.get();

}

public void setHandler(Object handler)

{

this.handlerRef = new WeakReference(handler);

}

@Override

public Object invoke(Object proxy, Method method, Object[] args)

throws Throwable

{

Object handler = handlerRef.get();

if (handler != null)

{

String methodName = method.getName();

method = methodMap.get(methodName);

if (method != null)

{

return method.invoke(handler, args);

}

}

return null;

}

}

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

效果图:

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

3、关于代理

======

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

//通过InvocationHandler设置代理

DynamicHandler handler = new DynamicHandler(activity);

handler.addMethod(methodName, method);

Object listener = Proxy.newProxyInstance(

listenerType.getClassLoader(),

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

package com.zhy.invocationhandler;

public class Button

{

private OnClickListener listener;

public void setOnClickLisntener(OnClickListener listener)

{

this.listener = listener;

}

public void click()

{

if (listener != null)

{

listener.onClick();

}

}

}

OnClickListener接口

package com.zhy.invocationhandler;

public interface OnClickListener

{

void onClick();

}

OnClickListenerHandler , InvocationHandler的实现类

package com.zhy.invocationhandler;

import java.lang.reflect.InvocationHandler;

import java.lang.reflect.Method;

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

最后

Android学习是一条漫长的道路,我们要学习的东西不仅仅只有表面的 技术,还要深入底层,弄明白下面的 原理,只有这样,我们才能够提高自己的竞争力,在当今这个竞争激烈的世界里立足。

人生不可能一帆风顺,有高峰自然有低谷,要相信,那些打不倒我们的,终将使我们更强大,要做自己的摆渡人。

资源持续更新中,欢迎大家一起学习和探讨。

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

又不知道该从何学起的朋友,同时减轻大家的负担。**

[外链图片转存中…(img-EgJyXvwa-1713769776767)]

[外链图片转存中…(img-vjcmIXDu-1713769776768)]

[外链图片转存中…(img-nStuHrzI-1713769776769)]

[外链图片转存中…(img-u5S7jLJL-1713769776770)]

[外链图片转存中…(img-UoI3LLS1-1713769776771)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

[外链图片转存中…(img-Dct395on-1713769776772)]

最后

Android学习是一条漫长的道路,我们要学习的东西不仅仅只有表面的 技术,还要深入底层,弄明白下面的 原理,只有这样,我们才能够提高自己的竞争力,在当今这个竞争激烈的世界里立足。

人生不可能一帆风顺,有高峰自然有低谷,要相信,那些打不倒我们的,终将使我们更强大,要做自己的摆渡人。

资源持续更新中,欢迎大家一起学习和探讨。

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

  • 4
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值