在Android中使用IOC框架让代码更清爽(二)

简介: 在Android中使用IOC框架让代码更清爽(二)

3.给按钮设置点击监听事件


平常我们给按钮设置监听事件的代码如下:

this.myBut2.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View v) {
    }
});

马上颠覆你的所见,步骤如下:


①创建注解类InjectOnClick。


代码如下:

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface InjectOnClick {
    int[] value();
}


你会发现,少了什么对吗?没错,接口类型,需要实现的方法,以及要调用设置的方法,这就需要下面第二步。


②设置注解类的注解类OnClickEvent。


代码如下:

@Target(ElementType.ANNOTATION_TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface OnClickEvent {
    Class<?> listenerType();//接口类型
    String listenerSetter();//设置的方法
    String methodName();//接口里面要实现的方法
}

修改InjectOnClick:


@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@OnClickEvent(listenerType = View.OnClickListener.class,listenerSetter = "setOnClickListener",methodName = "onClick")
public @interface InjectOnClick {
    int[] value();
}


③在MainActivity里设置注解。


代码如下:

@InjectOnClick({R.id.myBut1,R.id.myBut2})
public void setButtonOnClickListener(View view){
    switch (view.getId()){
        case R.id.myBut1:
            Toast.makeText(this,"你好",Toast.LENGTH_SHORT).show();
            break;
        case R.id.myBut2:
            Toast.makeText(this,"我很帅",Toast.LENGTH_SHORT).show();
            break;
        default:
            break;
    }
}


④注入点击事件


代码在InjectUtils中,如下:

/**
 * 注入点击事件
 * @param activity
 */
public static void injectOnClickListener(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();//获取具体的注解类
            OnClickEvent onClickEvent = annotationType.getAnnotation(OnClickEvent.class);//取出注解的onClickEvent注解
            if(onClickEvent!=null){//如果不为空
                try {
                    Method valueMethod=annotationType.getDeclaredMethod("value");//获取注解InjectOnClick的value方法
                    int[] viewIds= (int[]) valueMethod.invoke(annotation,null);//获取控件值
                    Class<?> listenerType = onClickEvent.listenerType();//获取接口类型
                    String listenerSetter = onClickEvent.listenerSetter();//获取set方法
                    String methodName = onClickEvent.methodName();//获取接口需要实现的方法
                    MyInvocationHandler handler=new MyInvocationHandler(activity);//自己实现的代码,负责调用
                    handler.setMethodMap(methodName,method);//设置方法及设置方法
                    Object object= Proxy.newProxyInstance(listenerType.getClassLoader(),new Class<?>[]{listenerType},handler);//创建动态代理对象类
                    for (int viewId:viewIds){//遍历要设置监听的控件
                        View view=activity.findViewById(viewId);//获取该控件
                        Method m=view.getClass().getMethod(listenerSetter, listenerType);//获取方法
                        m.invoke(view,object);//调用方法
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }
    }
}


虽然注解很详细,但还是有必要说明一下:


㈠关于因为我们用的是Java的动态代理,每一个动态代理类都必须要实现InvocationHandler这个接口,并且每个代理类的实例都关联到了一个handler,当我们通过代理对象调用一个方法的时候,这个方法的调用就会被转发为由InvocationHandler这个接口的 invoke 方法来进行调用。我们来看看InvocationHandler这个接口的唯一一个方法invoke方法:


Object invoke(Object proxy, Method method, Object[] args) throws Throwable


proxy:  指代我们所代理的那个真实对象

method:  指代的是我们所要调用真实对象的某个方法的Method对象

args:  指代的是调用真实对象某个方法时接受的参数


㈡Proxy这个类的作用就是用来动态创建一个代理对象的类,它提供了许多的方法,但是我们用的最多的就是newProxyInstance这个方法:


public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) throws IllegalArgumentException

loader:  一个ClassLoader对象,定义了由哪个ClassLoader对象来对生成的代理对象进行加载


interfaces:  一个Interface对象的数组,表示的是我将要给我需要代理的对象提供一组什么接口,如果我提供了一组接口给它,那么这个代理对象就宣称实现了该接口(多态),这样我就能调用这组接口中的方法了


h:  一个InvocationHandler对象,表示的是当我这个动态代理对象在调用方法的时候,会关联到哪一个InvocationHandler对象上


㈢getFields()与getDeclaredFields()区别:getFields()只能访问类中声明为公有的字段,私有的字段它无法访问,能访问从其它类继承来的公有方法.getDeclaredFields()能访问类中所有的字段,与public,private,protect无关,不能访问从其它类继承来的方法


getMethods()与getDeclaredMethods()区别:getMethods()只能访问类中声明为公有的方法,私有的方法它无法访问,能访问从其它类继承来的公有方法.getDeclaredFields()能访问类中所有的字段,与public,private,protect无关,不能访问从其它类继承来的方法


我们实现的InvocationHandler接口,代码如下:

public class MyInvocationHandler implements InvocationHandler {
    private Object object;
    private Map<String, Method> methodMap = new HashMap<String, Method>(1);
    public MyInvocationHandler(Object object) {
        this.object = object;
    }
    public void setMethodMap(String name, Method method) {
        this.methodMap.put(name, method);
    }
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        if (object != null) {
            String name = method.getName();
            method= this.methodMap.get(name);
            if (method != null) {
                return method.invoke(object, args);
            }
        }
        return null;
    }
}


这就不用我过多的注释了,因为,大部分代码与前面的并无不同,代理顾名思义就是帮你调用一些方法。


其代码难点就在这个代理里面,其他的代码与上面获取布局,控件一样,唯一不同的是这个代码用到了注释的注释,嵌套层级有点多,所以用到了许两层For循环,时间复杂度可能增加一个数量级,但这在手机上也是可以忽略不计的。


把方法加入InjectUtils的injectAll中。

/**
 * 注入所有
 * @param activity
 */
public static void injectAll(Activity activity){
    injectContentView(activity);
    injectControl(activity);
    injectOnClickListener(activity);
}

运行效果如下:

6.png


4.看一下代码有多简洁


代码如下:

@ContentView(R.layout.activity_main)
public class MainActivity extends AppCompatActivity {
    @InjectControl(value = R.id.content)
    private TextView content;
    @InjectControl(value = R.id.myBut1)
    private Button myBut1;
    @InjectControl(value = R.id.myBut2)
    private Button myBut2;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        InjectUtils.injectAll(MainActivity.this);
    }
    @InjectOnClick({R.id.myBut1,R.id.myBut2})
    public void setButtonOnClickListener(View view){
        switch (view.getId()){
            case R.id.myBut1:
                Toast.makeText(this,"你好",Toast.LENGTH_SHORT).show();
                break;
            case R.id.myBut2:
                Toast.makeText(this,"我很帅",Toast.LENGTH_SHORT).show();
                break;
            default:
                break;
        }
    }
}


可能有的人会问,我用一句findViewById或者一句setOnClickListener就可以解决的事,没事给自己找这复杂的代码写,我是不是有病啊。可是你忽略了一点,一但我将这个代码打包,这代码就可能复用到我所有的项目中,如果,你只开发小项目或一个项目,这样写确实不划算,但是如果你总是换项目就算不换,增加新界面的时候,总是需要写findViewById,那么当累计的一定的数量时,这样写是一定节省很多时间的。

相关文章
|
1月前
|
SQL 缓存 安全
Android ORM 框架之 greenDAO
Android ORM 框架之 greenDAO
37 0
|
2月前
|
Ubuntu 网络协议 Java
【Android平板编程】远程Ubuntu服务器code-server编程写代码
【Android平板编程】远程Ubuntu服务器code-server编程写代码
|
3天前
|
Ubuntu Android开发 数据安全/隐私保护
【Android平板编程】远程Ubuntu服务器Code-Server编程写代码
【Android平板编程】远程Ubuntu服务器Code-Server编程写代码
|
4天前
|
Java Android开发
Android 长按桌面显示菜单的代码
Android 长按桌面显示菜单的代码
9 0
|
4天前
|
Java Android开发
Android 切换壁纸代码流程追踪
Android 切换壁纸代码流程追踪
13 0
|
4天前
|
编解码 调度 Android开发
Android音频框架之一 详解audioPolicy流程及HAL驱动加载与配置
Android音频框架之一 详解audioPolicy流程及HAL驱动加载与配置
11 0
|
4天前
|
Shell Android开发
Android Activity重写dump方法实现通过adb调试代码
Android Activity重写dump方法实现通过adb调试代码
11 0
|
4天前
|
Java API Android开发
Android系统升级A/B分区OTA升级应用层调用UpdateEngine代码
Android系统升级A/B分区OTA升级应用层调用UpdateEngine代码
9 0
|
2月前
|
Android开发
android studio 重新将module中的代码加入到自己项目中,报错找不到SO文件。
android studio 重新将module中的代码加入到自己项目中,报错找不到SO文件。
10 1
|
3月前
|
Ubuntu 网络协议 Linux
【Linux】Android平板上远程连接Ubuntu服务器code-server进行代码开发
【Linux】Android平板上远程连接Ubuntu服务器code-server进行代码开发
65 0