android使用篇(三)
MVC模式
中提到一个问题:

1) 视图层(View):一般採用XML文件进行界面的描写叙述,使用的时候能够很方便的引入,可是用xml编写了,又须要在Acitvity声明而且实例化,有点麻烦,考虑是否能做一个类似注解实现匹配,或者写一个类获取xml的各个节点然后自己主动进行封装,当然,这仅仅是个想法,以后再实现。

今天最终把这个想法实现了,使用依赖注入IOC注解实现对activity中控件的实例化。

先普及一下java的反射机制和注解机制的知识:

下面引用大神的两篇文章:

JAVA反射机制


java 注解

完毕后仅仅须要: @ViewInject(id=R.id.btn1,click="btnClick") TextView btn1;  就可以完毕实例化,并加入�点击事件

基本思路:

一,public abstract class D3Activity extends Activity    写一个类继承Activity。

二,重写  setContentView   在此方法实现注解。

三,Field[] fields = activity.getClass().getDeclaredFields();    获取activity中的字段属性

四, field.getAnnotation(ViewInject.class);         获取字段的注解属性

五, field.set(activity,sourceView.findViewById(viewId));       实例化控件

大功告成,到此已实现了注解实现对android中activity和xml文件的实例化问题。

另外也能够实现注解对控件的事件加入�,具体

分三个类实现:

实现注解类:

注解类可注入 id---相应xml的id,各种点击事件,可自定义

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface D3View {
 public int id() default 0;
 public String click() default "";
 public String longClick() default "";
 public String itemClick() default "";
 public String itemLongClick() default "";
}

重写activity类,使用反射和注解实现实例化并添�事件:


public abstract class D3Activity extends Activity {

public void setContentView(int layoutResID) {
  super.setContentView(layoutResID);
  initInjectedView(this);
 } public void setContentView(View view, LayoutParams params) {
  super.setContentView(view, params);
  initInjectedView(this);
 } public void setContentView(View view) {
  super.setContentView(view);
  initInjectedView(this);
 } private void initInjectedView(Activity activity){
  initInjectedView(activity, activity.getWindow().getDecorView());
 }
 
 
 private void initInjectedView(Object activity,View sourceView){
  Field[] fields = activity.getClass().getDeclaredFields();   //获取字段
  if(fields!=null && fields.length>0){
   for(Field field : fields){
    try {
     field.setAccessible(true);   //设为可訪问
     
     if(field.get(activity)!= null )
      continue;
    
     D3View d3View = field.getAnnotation(D3View.class);
     if(d3View!=null){
      
      int viewId = d3View.id();
      if(viewId == 0)
       viewId = getResources().getIdentifier(field.getName(), "id",getPackageName());
      if(viewId == 0)
       Log.e("D3Activity", "field "+ field.getName() + "not found");
      
      //关键,注解初始化,相当于 backBtn = (TextView) findViewById(R.id.back_btn);
         field.set(activity,sourceView.findViewById(viewId)); 
         //事件
         setListener(activity,field,d3View.click(),Method.Click);
      setListener(activity,field,d3View.longClick(),Method.LongClick);
      setListener(activity,field,d3View.itemClick(),Method.ItemClick);
      setListener(activity,field,d3View.itemLongClick(),Method.itemLongClick);
     }
    } catch (Exception e) {
     e.printStackTrace();
    }
   }
  }
 }
 
 private void setListener(Object activity,Field field,String methodName,Method method)throws Exception{
  if(methodName == null || methodName.trim().length() == 0)
   return;
  
  Object obj = field.get(activity);
  
  switch (method) {
   case Click:
    if(obj instanceof View){
     ((View)obj).setOnClickListener(new EventListener(activity).click(methodName));
    }
    break;
   case ItemClick:
    if(obj instanceof AbsListView){
     ((AbsListView)obj).setOnItemClickListener(new EventListener(activity).itemClick(methodName));
    }
    break;
   case LongClick:
    if(obj instanceof View){
     ((View)obj).setOnLongClickListener(new EventListener(activity).longClick(methodName));
    }
    break;
   case itemLongClick:
    if(obj instanceof AbsListView){
     ((AbsListView)obj).setOnItemLongClickListener(new EventListener(activity).itemLongClick(methodName));
    }
    break;
   default:
    break;
  }
 }
 
 public enum Method{
  Click,LongClick,ItemClick,itemLongClick
 }
 
}

事件类: 实现了 OnClickListener, OnLongClickListener, OnItemClickListener,OnItemLongClickListener ,能够自己扩展

public class EventListener implements OnClickListener, OnLongClickListener, OnItemClickListener,OnItemLongClickListener {

	private Object handler;

	private String clickMethod;
private String longClickMethod;
private String itemClickMethod;
private String itemLongClickMehtod; public EventListener(Object handler) {
this.handler = handler;
} public EventListener click(String method){
this.clickMethod = method;
return this;
} public EventListener longClick(String method){
this.longClickMethod = method;
return this;
} public EventListener itemLongClick(String method){
this.itemLongClickMehtod = method;
return this;
} public EventListener itemClick(String method){
this.itemClickMethod = method;
return this;
} public boolean onLongClick(View v) {
return invokeLongClickMethod(handler,longClickMethod,v);
} public boolean onItemLongClick(AdapterView<?> arg0, View arg1, int arg2,long arg3) {
return invokeItemLongClickMethod(handler,itemLongClickMehtod,arg0,arg1,arg2,arg3);
} public void onItemClick(AdapterView<?> arg0, View arg1, int arg2, long arg3) { invokeItemClickMethod(handler,itemClickMethod,arg0,arg1,arg2,arg3);
} public void onClick(View v) { invokeClickMethod(handler, clickMethod, v);
} private static Object invokeClickMethod(Object handler, String methodName, Object... params){
if(handler == null) return null;
Method method = null;
try{
method = handler.getClass().getDeclaredMethod(methodName,View.class);
if(method!=null)
return method.invoke(handler, params);
else
throw new RuntimeException("no such method:"+methodName);
}catch(Exception e){
e.printStackTrace();
} return null; } private static boolean invokeLongClickMethod(Object handler, String methodName, Object... params){
if(handler == null) return false;
Method method = null;
try{
//public boolean onLongClick(View v)
method = handler.getClass().getDeclaredMethod(methodName,View.class);
if(method!=null){
Object obj = method.invoke(handler, params);
return obj==null?false:Boolean.valueOf(obj.toString());
}
else
throw new RuntimeException("no such method:"+methodName);
}catch(Exception e){
e.printStackTrace();
} return false; } private static Object invokeItemClickMethod(Object handler, String methodName, Object... params){
if(handler == null) return null;
Method method = null;
try{
///onItemClick(AdapterView<?> arg0, View arg1, int arg2, long arg3)
method = handler.getClass().getDeclaredMethod(methodName,AdapterView.class,View.class,int.class,long.class);
if(method!=null)
return method.invoke(handler, params);
else
throw new RuntimeException("no such method:"+methodName);
}catch(Exception e){
e.printStackTrace();
} return null;
} private static boolean invokeItemLongClickMethod(Object handler, String methodName, Object... params){
if(handler == null) throw new RuntimeException("invokeItemLongClickMethod: handler is null :");
Method method = null;
try{
///onItemLongClick(AdapterView<?> arg0, View arg1, int arg2,long arg3)
method = handler.getClass().getDeclaredMethod(methodName,AdapterView.class,View.class,int.class,long.class);
if(method!=null){
Object obj = method.invoke(handler, params);
return Boolean.valueOf(obj==null?false:Boolean.valueOf(obj.toString()));
}
else
throw new RuntimeException("no such method:"+methodName);
}catch(Exception e){
e.printStackTrace();
} return false;
}
}

到此已经完毕了,仅仅须要这样就可以实例化:

public class MainActivity extends D3Activity {

	//@ViewInject EditText input;   //id和属性名同样,自己主动匹配
@ViewInject(id = R.id.input) EditText editText;
@ViewInject(click="btnClick") TextView btn1,btn2,btn3; public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
} public void btnClick(View v){ switch (v.getId()) {
case R.id.btn1:
btn1.setText(editText.getText().toString());
Toast.makeText(getApplicationContext(), "111", Toast.LENGTH_SHORT).show();
break; case R.id.btn2:
Toast.makeText(getApplicationContext(), "222", Toast.LENGTH_SHORT).show(); break; case R.id.btn3:
Toast.makeText(getApplicationContext(), "333", Toast.LENGTH_SHORT).show();
break; default:
break;
} } }

相应xml布局文件:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="vertical" > <EditText
android:id="@+id/input"
android:layout_width="fill_parent"
android:layout_height="wrap_content" />
<TextView
android:id="@+id/btn1"
android:layout_marginTop="10dp"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="设置自己"
/>
<TextView
android:id="@+id/btn2"
android:layout_marginTop="10dp"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="btn2"
/> <TextView
android:id="@+id/btn3"
android:layout_marginTop="10dp"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="btn3"
/>
</LinearLayout>

源代码已经放在了github,有兴趣的能够去看看 :https://github.com/mozhenhau/injectAndroid.git

android使用篇(四) 注解依赖注入IOC实现绑定控件的更多相关文章

  1. 转:深入浅出spring IOC中四种依赖注入方式

    转:https://blog.csdn.net/u010800201/article/details/72674420 深入浅出spring IOC中四种依赖注入方式 PS:前三种是我转载的,第四种是 ...

  2. [Android]使用Dagger 2进行依赖注入 - Producers(翻译)

    以下内容为原创,欢迎转载,转载请注明 来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/6234811.html 使用Dagger 2进行依赖注入 - P ...

  3. 从壹开始前后端分离【 .NET Core2.0 +Vue2.0 】框架之九 || 依赖注入IoC学习 + AOP界面编程初探

    更新 1.如果看不懂本文,或者比较困难,先别着急问问题,我单写了一个关于依赖注入的小Demo,可以下载看看,多思考思考注入的原理: https://github.com/anjoy8/BlogArti ...

  4. Z从壹开始前后端分离【 .NET Core2.2/3.0 +Vue2.0 】框架之九 || 依赖注入IoC学习 + AOP界面编程初探

    本文梯子 本文3.0版本文章 更新 代码已上传Github+Gitee,文末有地址 零.今天完成的绿色部分 一.依赖注入的理解和思考 二.常见的IoC框架有哪些 1.Autofac+原生 2.三种注入 ...

  5. spring四种依赖注入方式(转)

    spring四种依赖注入方式!! 平常的java开发中,程序员在某个类中需要依赖其它类的方法,则通常是new一个依赖类再调用类实例的方法,这种开发存在的问题是new的类实例不好统一管理,spring提 ...

  6. WPF 高级篇 MVVM (MVVMlight) 依赖注入使用Messagebox

    原文:WPF 高级篇 MVVM (MVVMlight) 依赖注入使用Messagebox MVVMlight 实现依赖注入 把弹框功能 和接口功能注入到各个插件中 使用依赖注入 先把所有的ViewMo ...

  7. Android 高级UI设计笔记11:Gallery(画廊控件)之Gallery基本使用

    1. 这里要向大家介绍Android控件Gallery(画廊控件) Gallery控件主要用于横向显示图像列表,不过按常规做法.Gallery组件只能有限地显示指定的图像.也就是说,如果为Galler ...

  8. Android开发系列(十八):自己定义控件样式在drawable目录下的XML实现

    在Android开发的过程中,我们常常须要对控件的样式做一下改变,能够通过用添加背景图片的方式进行改变,可是背景图片放多了肯定会使得APK文件变的非常大. 我们能够用自己定义属性shape来实现. s ...

  9. 将四个按钮放入一个父控件的好处:方便移动,只需要改变父控件的y值,就可移动四个按钮

      将四个按钮放入一个父控件的好处:方便移动,只需要改变父控件的y值, 就可移动四个按钮               https://www.evernote.com/shard/s227/sh/78 ...

随机推荐

  1. sql基础,必须会的————随便整理、杂乱无章

    1.sqlserver2008r2的安装 2.数据库与表的建立.增加.删除.修改. 3,索引的概念,包括聚集与非聚集的区别.全文索引的建立与如何使用全文索引. 4,重新生成索引,重新组织索引. 5,建 ...

  2. JavaScript 字符串编码函数

    JavaScript中有三个可以对字符串编码的函数,分别是: escape,encodeURI,encodeURIComponent,相应3个解码函数:unescape,decodeURI,decod ...

  3. Python 第七篇:socket编程

    一:socket基础: 1.1:Socket基础: socket起源于Unix,而Unix/Linux基本哲学之一就是“一切皆文件”,对于文件用[打开][读写][关闭]模式来操作.socket就是该模 ...

  4. 墙内安装nautilus-dropbox 1.6.0-2

    软件包:nautilus-dropbox 版本号:1.6.0-2 # aptitude install nautilus-dropbox # less `which dropbox` 查找到下载链接: ...

  5. 面试中的Singleton

      引子 “请写一个Singleton.”面试官微笑着和我说. “这可真简单.”我心里想着,并在白板上写下了下面的Singleton实现: 1 class Singleton 2 { 3 public ...

  6. cocos2d-x游戏开发系列教程-中国象棋00-前言

    象棋描述 在说代码之前,我们先让象棋效果登场,以方便大家对代码的理解 欢迎界面 中国象棋程序,运行起来的第一个界面是一个欢迎界面,该欢迎界面在停留一秒后进入游戏界面 游戏主界面 新局:所有棋子归位,状 ...

  7. Ajax技术——带进度条的文件上传

    1.概述 在实际的Web应该开发或网站开发过程中,经常需要实现文件上传的功能.在文件上传过程中,经常需要用户进行长时间的等待,为了让用户及时了解上传进度,可以在上传文件的同时,显示文件的上传进度条.运 ...

  8. SilkTest Q&A 12

    111. 谁能告诉我,正在执行的SilkTest的log是存放在哪里? 答案1: 用下面的命令可以导出文本格式的log "c:/program files/segue/silktest/pa ...

  9. MySQL存储引擎:InnoDB和MyISAM的差别/优劣评价/评测/性能测试

    InnoDB和MyISAM简介 MyISAM:这个是默认类型,它是基于传统的ISAM类型,ISAM是Indexed Sequential Access Method (有索引的 顺序访问方法) 的缩写 ...

  10. 推荐国内外优秀+免费CDN加速站点及公共cdn加速库

    -----------------------------------------------------------------免费CDN加速站点 1.CloudFlare CloudFlare可能 ...