android(基于监听)的事件处理
基于监听的事件模型分工更明确,事件源、事件监听由两个类分开,因此具有更好的可维护性。
Android的事件处理机制保证基于监听的事件监听器会被优先出发。
在事件监听的处理模型中,主要涉及三类对象:
1.Event Source(事件源):事件发生的场所,通常就是各个组件,例如按钮,窗口,菜单等。
2.Event(事件):事件封装了界面组件上发生的特定事情(通常就是一次用户操作)。如果程序需要获得界面组件上所发生事情的相关信息,一般通过Event对象来取得。
3.Event Listener(事件监听器):负责监听事件源所发生的事件,并对各种事件做出相应的响应。
Android为不同的界面组件提供了不同的监听器接口:
1.View.OnClickListener:单击事件的事件监听器必须实现的接口。
2.View.OnCreateContextMenuListener:创建上下文菜单事件的事件监听器必须实现的接口。
3.View.onFocusChangeListener:焦点改变事件的事件监听器必须实现的接口。
4.View.OnKeyListener:按键事件的事件监听器必须实现的接口。
5.View.OnLongClickListener:长按事件的事件监听器必须实现的接口。
6.View.OnTouchListener:触摸事件的事件监听器必须实现的接口。
所谓的事件监听器,其实就是实现了特定接口的java类的实例。在程序中实现事件监听器,通常有如下几种形式。
1.内部类形式:将事件监听器类定义成当前的内部类。
2.外部类形式:将事件监听器类定义成一个外部类。
3.Activity本身作为事件监听器类:让Activity本身实现监听器接口,并实现事件处理方法。
4.匿名内部类形式:使用匿名内部类创建事件监听器对象。
5.直接绑定标签:为UI组件的android:onClick属性指定事件的监听方法,开发者需要在Activity中定义该事件监听方法(该方法必须有一个View类型的形参,该形参代表被单击的UI组件),当用户单击该UI组件时,系统将会激发android:onClick属性所指定的方法。
1.内部类作为事件监听器类
使用内部类作为事件监听器类的优势:
①使用内部类可以在当前类中复用该监听器类
②因为监听器类是外部类的内部类,所以可以自由访问外部类的所有界面组件。
示例:
activity_main.xml
<LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"><Buttonandroid:id="@+id/bn"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="内部类做为事件监听器类"/></LinearLayout>
MainActivity.java
packagecom.example.neibulei;publicclassMainActivityextendsActivity{privateButtonbn;@OverrideprotectedvoidonCreate(BundlesavedInstanceState){super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);//获取activity_main界面中的id为bn按钮bn=(Button)findViewById(R.id.bn);//为bn按钮绑定点击事件监听器bn.setOnClickListener(newMyClickListener());}//定义一个单击事件监听器classMyClickListenerimplementsView.OnClickListener{@Override//实现监听器类必须实现的方法,该方法将会作为事件处理器publicvoidonClick(Viewv){//TODOAuto-generatedmethodstub//事件处理器做出的响应Toast.makeText(MainActivity.this,"内部类做为事件监听器类",1).show();}}}
2.外部类作为事件监听器类
外部类作为事件监听器类的劣势:
①事件监听器通常属于特定的GUI界面,定义成外部类不利于提高程序的内聚性。
②外部类形式的事件监听器不能自由访问GUI界面的类中的组件,变成不够简洁。
外部类作为事件监听器类的优势:
如果某个事件监听器确实需要被多个GUI界面所共享,而且主要是完成某种业务逻辑的实现,则可以考虑使用外部类的形式来定义事件监听器类。
示例:
activity_main.xml
<LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"><EditTextandroid:id="@+id/address"android:layout_width="fill_parent"android:layout_height="wrap_content"/><EditTextandroid:id="@+id/content"android:layout_width="fill_parent"android:layout_height="wrap_content"/><Buttonandroid:id="@+id/bn"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="外部类作为事件监听器类(发送短信)"/></LinearLayout>
MainActivity.java
packagecom.example.waibulei;publicclassMainActivityextendsActivity{EditTextaddress;//短信发送给这个号码EditTextcontent;//短信内容Buttonbn;//发送按钮@OverrideprotectedvoidonCreate(BundlesavedInstanceState){super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);//获取页面内中的收件人地址、短信内容address=(EditText)findViewById(R.id.address);content=(EditText)findViewById(R.id.content);bn=(Button)findViewById(R.id.bn);//为bn按钮绑定长按事件监听器bn.setOnLongClickListener(newSendSmsListener(this,address,content));}}
SendSmsListener.java
packagecom.example.waibulei;publicclassSendSmsListenerimplementsOnLongClickListener{privateActivityact;privateEditTextaddress;//短信发送给这个号码privateEditTextcontent;//短信内容//构造方法取得上下文环境,获得两个文本框控件publicSendSmsListener(Activityact,EditTextaddress,EditTextcontent){this.act=act;this.address=address;this.content=content;}@OverridepublicbooleanonLongClick(Viewv){//取出号码与短信内容StringaddressStr=address.getText().toString().trim();StringcontentStr=content.getText().toString().trim();//获取短信管理器SmsManagersmsManager=SmsManager.getDefault();//发送文本短信smsManager.sendTextMessage(addressStr,null,contentStr,null,null);returnfalse;}}
在AndroidManifest.xml上添加发送短信的权限
<uses-permissionandroid:name="android.permission.SEND_SMS"/>
3.Activity本身作为事件监听器
Activity本身作为事件监听器的劣势:
①这种形式可能造成程序结构混乱,Activity的主要职责应该是完成界面初始化工作,但此时还需要包含事件处理器的方法,从而引起混乱。
②如果Activity界面类需要实现监听器接口,让人感觉比较怪异。
Activity本身作为事件监听器的优势:
直接在Activity类中定义事件处理方法,非常简洁。
示例:
activity_main.xml
<LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"><Buttonandroid:id="@+id/bn"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="Activity本身作为事件监听器"/></LinearLayout>
MainActivity.java
packagecom.example.activitybenshen;//实现事件监听器接口publicclassMainActivityextendsActivityimplementsOnClickListener{privateButtonbn;@OverrideprotectedvoidonCreate(BundlesavedInstanceState){super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);bn=(Button)findViewById(R.id.bn);//直接使用Activity作为监听器bn.setOnClickListener(this);}@Override//实现事件处理方法publicvoidonClick(Viewv){//TODOAuto-generatedmethodstubToast.makeText(MainActivity.this,"直接使用Activity作为监听器",1).show();}}
4.匿名内部类作为事件监听器类
匿名内部类作为事件监听器类的优势:
大部分时候,事件处理器都没有什么复用价值(可复用代码通常都被抽象成了业务逻辑方法),因此大部分事件监听器只是临时使用一次,所以使用匿名内部类形式的事件监听器更合适。
匿名内部类作为事件监听器类的劣势:
语法不宜掌握。
示例:
activity_main.xml
<LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"><Buttonandroid:id="@+id/bn"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="匿名内部类作为事件监听器类"/></LinearLayout>
MainActivity.java
packagecom.example.nimingneibulei;publicclassMainActivityextendsActivity{privateButtonbn;@OverrideprotectedvoidonCreate(BundlesavedInstanceState){super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);bn=(Button)findViewById(R.id.bn);//使用匿名内部类作为事件监听器bn.setOnClickListener(newOnClickListener(){@OverridepublicvoidonClick(Viewv){//事件监听器的响应Toast.makeText(MainActivity.this,"匿名内部类作为事件监听器",1).show();}});}}
5.直接绑定到标签
对于很多Android界面组件标签而言,它们都支持OnClick属性,该属性值就是一个形如xxx(View source)的方法的方法名。
示例:
activity_main.xml
<LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"><Buttonandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:onClick="clickHandler"android:text="直接绑定到标签onClick"/></LinearLayout>
MainActivity.java
packagecom.example.onclick;publicclassMainActivityextendsActivity{@OverrideprotectedvoidonCreate(BundlesavedInstanceState){super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);}//定义一个事件处理方法//其中source参数代表事件源publicvoidclickHandler(Viewsource){Toast.makeText(MainActivity.this,"直接绑定到标签onClick",1).show();}}
声明:本站所有文章资源内容,如无特殊说明或标注,均为采集网络资源。如若本站内容侵犯了原著者的合法权益,可联系本站删除。