首页 > 代码库 > 来,咱们自己写一个Android的IOC框架!

来,咱们自己写一个Android的IOC框架!

到目前位置,afinal开发框架也是用了好几个月了,还记得第一次使用注释完成控件的初始化和事件绑定的时候,当时的心情是多么的兴奋— —代码竟然可以这样写!然后随着不断的学习,也慢慢的对IOC框架和注解反射等东西有了一点简单的了解,之前的一篇文章简单的介绍了一下Java的反射机制,今天的文章,就完成一个简单的,基于IOC的小Demo,让大家慢慢的对IOC有一点简单的了解。

首先,什么是IOC呢?

控制反转(Inversion of Control,英文缩写为IoC)是一个重要的面向对象编程的法则来削减计算机程序的耦合问题,也是轻量级的Spring框架的核心。 控制反转一般分为两种类型,依赖注入(Dependency Injection,简称DI)和依赖查找。依赖注入应用比较广泛。

我们下面要完成的,就是Android中依赖注入的实现。

首先,看我们的项目结构


结构很简单,一个基类,一个子类,一个自定义注释类型,一个布局文件。

下面看代码的具体实现

首先看最重要的基类

BaseActivity.java

package com.example.iocdemo;

import java.lang.reflect.Field;

import android.app.Activity;
import android.content.Context;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Toast;

public class BaseActivity extends Activity {

	protected Context mContext = this;

	/**
	 * 实现IOC注入
	 * 
	 * @param baseActivity
	 */
	public void initInjectedView(Object baseActivity) {
		// 获取所有的成员变量
		Field[] fields = baseActivity.getClass().getDeclaredFields();
		if (fields != null && fields.length > 0) {
			// 遍历成员变量
			for (Field field : fields) {
				try {
					// 抑制权限检查
					field.setAccessible(true);
					// 获取成员变量的注释类
					ViewInject viewInject = field
							.getAnnotation(ViewInject.class);
					// 如果注释类不为空,即成员变量是使用注释的方式进行声明的
					if (viewInject != null) {
						// 获取注释中的id
						int id = viewInject.id();
						// 设置字段值
						field.set(this,
								((Activity) baseActivity).findViewById(id));
						// 将实例化好的View对象取出
						View view = (View) field.get(this);
						// 绑定监听事件
						view.setOnClickListener(new OnClickListener() {

							@Override
							public void onClick(View v) {
								Toast.makeText(mContext, "别点人家吖!",
										Toast.LENGTH_SHORT).show();
							}
						});
					}

				} catch (Exception e) {
					e.printStackTrace();
				}
			}
		}
	}
}

在这个类里面,我们完成了IOC注入方法的编写,自定义的注释类型代码如下:

package com.example.iocdemo;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface ViewInject {
	public int id();
}
@Target(ElementType.FIELD)这句代码实现的是控制注释的位置为字段,或者说是成员变量,因为我们要完成的是控件的注释,和时间绑定,因此,我们设置为ElementType.FIELD就可以。

@Retention是一个enum类型,共有三个值,分别是SOURCE,CLASS 和 RUNTIME.

SOURCE代表的是这个Annotation类型的信息只会保留在程序源码里,源码如果经过了编译之后,Annotation的数据就会消失,并不会保留在编译好的.class文件里面

ClASS的意思是这个Annotation类型的信息保留在程序源码里,同时也会保留在编译好的.class文件里面,在执行的时候,并不会把这一些信息加载到虚拟机(JVM)中去.注意一下,当你没有设定一个Annotation类型的Retention值时,系统默认值是CLASS. 
RUNTIME,表示在源码、编译好的.class文件中保留信息,在执行的时候会把这一些信息加载到JVM中去的

因为我们需要在JVM把我们的class文件加载进入之后,完成注入,因此,我们选择这个属性。

好了,现在我们知道如何简单的定义一个自定义的注释类型,并且用代码实现了代码注入和事件绑定,下面,我们看一下在我们的程序中,如何使用。

下面是我们在Activity的代码

package com.example.iocdemo;

import android.os.Bundle;
import android.widget.Button;

public class MainActivity extends BaseActivity {

	//用注释进行控件的初始化
	@ViewInject(id = R.id.btn)
	Button btn;

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);
		//进行注入
		initInjectedView(this);
	}

}

下面是我们的运行结果