对于Android提供的事件处理模型,不难发现基于监听的事件处理模型具有更大的优势:
- 基于监听的事件模型分工更加明确,事件源、事件监听有两个类分开实现,因此具有更好的维护性。
- Android的事件处理机制保证基于基于监听的事件监听器会被优先触发。
实例:通过回调实现跟随手指的小球
在某些特定情况下,基于回调的事件处理机制会更好地提高程序的内聚性,例如上一章的实例;跟随手指的小球,如果改为基于回调的实现,可以更好的提高程序的内聚性。
例如将该实例中的DrawView类改为如下形式。
package com.example.studyevent; import java.text.AttributedCharacterIterator.Attribute; import android.content.Context; import android.graphics.Canvas; import android.graphics.Color; import android.graphics.Paint; import android.util.AttributeSet; import android.view.MotionEvent; import android.view.View; public class DrawView extends View { public float currentX=40; public float currentY=50; //定义、创建画笔 Paint p=new Paint(); public DrawView(Context context,AttributeSet set) { super(context,set); // TODO Auto-generated constructor stub } @Override public void onDraw(Canvas canvas) { super.onDraw(canvas); //设置画笔的颜色 p.setColor(Color.RED); //绘制一个小球(作为小球) canvas.drawCircle(currentX, currentY, 15, p); } @Override public boolean onTouchEvent(MotionEvent event) { //当前组件的currentX、currentY两个属性 this.currentX=event.getX(); this.currentY=event.getY(); //通知该组件重绘 this.invalidate(); //返回true表明处理方法已经处理该事件 return true; } }
上面的程序中粗体字代码重写了View组件的onTouchEvent(MotionEvent event)方法,这表示该组件就可以处理触摸屏事件,当用户手指在屏幕上移动时,该View上绘制的小球将会跟随用户手指。也就是说,这个自定义View本身就可以很好的处理屏幕事件。
接下来在程序中使用这个DrawView几乎不需增加任何处理,直接将View放到界面布局中即可。如下面界面布局中的XML文件所示。
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" > <!-- 使用自定义组件 --> <com.example.studyevent.DrawView android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent"/> </LinearLayout>
接下来Activity类中不需要为该View绑定事件监听器——因为该View自己就可以处理它的触摸屏事件。
运行上面程序将看到下图所示效果:
通过为View提供事件处理的回调方法,可以很好地把事件处理方法封装在该View内部,从而提高程序的内聚性——基于回调的事件处理更适合于应付那种事件处理逻辑比价固定的View,比如上面介绍的这个跟随用户手指的View。