打造自己的IOC注解

在項(xiàng)目開發(fā)中,想要獲取布局中相應(yīng)的控件都是通過findViewById來實(shí)現(xiàn)的,當(dāng)然還可以使用一些第三方的注解框架,像XUtils、butterKnidnife等;本人的話不太喜歡使用這些第三方的,可能不太習(xí)慣還有就是改動(dòng)不太方便,所有在開發(fā)中都是在BaseActivity封裝一下,

   @SuppressWarnings("unchecked")
    protected <T extends View> T ID(int id) {
        return (T) super.findViewById(id);
    }

然后在子類中使用,這樣其實(shí)只不過不用每次findViewById的時(shí)候進(jìn)行強(qiáng)轉(zhuǎn),后面在學(xué)習(xí)和開發(fā)過程中,看到一位大牛自己寫了一個(gè)IOC的注解,跟著也學(xué)了下,感覺還不錯(cuò),它是通過反射和注解的方式來實(shí)現(xiàn),并且可以根據(jù)自己的需要很方便的進(jìn)行添加和修改,根據(jù)需要,新建了三個(gè)注解類,ViewById、OnClick、CheckNet,ViewById是控件,OnClick是點(diǎn)擊事件,CheckNet是檢測(cè)網(wǎng)絡(luò);
ViewById注解:

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface ViewById {
    int value();
}

@Target代表的是注解的位置,有四中類型:

ElementType.FIELD  代表的是在屬性上
ElementType.METHOD  代表的是在方法上
ElementType.TYPE  代表的是在類上
ElementType.CONSTRUCTOR  代表的是在構(gòu)造方法上

@Retention表示什么什么時(shí)候生效,用三種類型;

RetentionPolicy.RUNTIME  運(yùn)行時(shí)生效
RetentionPolicy.CLASS  編譯時(shí)生效
RetentionPolicy.SOURCE  源碼資源

OnClick和CheckNet的實(shí)現(xiàn)都是一樣的,OnClick改下@Target的位置就可以了,CheckNet的話不用傳遞參數(shù);
通過反射和注解的方式,調(diào)用的還是android系統(tǒng)的findViewById方法,在寫反射方法之前提供一個(gè)findViewById的幫助類;

public class ViewFinder {
    private Activity mActivity;
    private View mView;

    public ViewFinder(Activity activity) {
        this.mActivity = activity;
    }

    public ViewFinder(View view) {
        this.mView = view;
    }

    public View findViewById(int viewId) {
        return mActivity != null ? mActivity.findViewById(viewId) : mView.findViewById(viewId);
    }
}

該類提供了針對(duì)Activity 和View 兩種構(gòu)造方法,接下來在ViewUtils類中來實(shí)現(xiàn)具體的方法就可以了;ViewUtils類中也提供了三種方式的構(gòu)造方法;

  public static void inject(Activity activity) {
        inject(new ViewFinder(activity), activity);
    }

    public static void inject(View view) {
        inject(new ViewFinder(view), view);
    }

    public static void inject(View view, Object object) {
        inject(new ViewFinder(view), object);
    }

    public static void inject(ViewFinder finder, Object object) {
        injectFiled(finder, object);
        injectEvent(finder, object);
    }

先根據(jù)傳入進(jìn)來的Object類,通過反射獲取到該類所有的屬性,

//獲取類中所有的屬性
 Class<?> clazz = object.getClass();
//獲取所有屬性包括公有和私有
Field[] fields = clazz.getDeclaredFields();

循環(huán)遍歷該屬性數(shù)組,得到Object類的每一個(gè)屬性,判斷該屬性上面是否與ViewById的注解一致,如果一致就通過set();方法設(shè)置;

        //獲取ViewById中的value值
        for (Field field : fields) {
            ViewById viewById = field.getAnnotation(ViewById.class);
            if (viewById != null) {
                //獲取注解里面的id值
                int value = viewById.value();
                //findViewById找到view
                View view = finder.findViewById(value);
                if (view != null) {
                    //能夠修飾所有修飾符
                    field.setAccessible(true);
                    //動(dòng)態(tài)注入找到的view
                    try {
                        field.set(object, view);
                    } catch (IllegalAccessException e) {
                        e.printStackTrace();
                    }
                }
            }
        }

在遍歷獲取每個(gè)屬性的時(shí)候要field.setAccessible(true);設(shè)置為true允許訪問Object類的私有屬性,這樣findViewById就ok了,點(diǎn)擊事件的話跟findViewById是一樣的將屬性換成方法就可以了;在使用的時(shí)候首先要將ViewUtils類進(jìn)行初始化;

    @ViewById(R.id.tv)
    private TextView tv;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ViewUtils.inject(this);

        tv.setText("IOCTest");
    }

這樣就實(shí)現(xiàn)了,運(yùn)行下,效果如下:

device-2017-06-29-214102.gif

源碼地址:http://pan.baidu.com/s/1nv2Zw6l

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

推薦閱讀更多精彩內(nèi)容