在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