android.os.Handler是Android SDK中处理定时操作的核心类。通过Handler类,可以提交和处理一个Runnable对象。这个对象的run 方法可以立刻执行,也可以在指定时间之后执行(可以称为预约执行)。
handler类允许你发送消息和处理线程消息队列中的消息及runnable对象。handler实例都是与一个线程和该线程的消息队列一起使用,一旦创建了一个新的handler实例,系统就把该实例与一个线程和该线程的消息队列捆绑起来,这将可以发送消息和runnable对象给该消息队列,并在消息队列出口处处理它们。
handler类有两种主要用途:
1、按照时间计划,在未来某时刻,对处理一个消息或执行某个runnable实例。
2、把一个对另外线程对象的操作请求放入消息队列中,从而避免线程间冲突。
时间类消息通过如下方法使用:
post(Runnable)
postAtTime(Runnable, long)
postDelayed(Runnable, long)
sendEmptyMessage(int)
sendMessage(Message)
sendMessageAtTime(Message, long)
andsendMessageDelayed(Message, long)
methods. post之类函数可以传输一个runnable对象给消息队列,并在到达消息队列后被调用。sendmessage之类函数可以传送一个包含数据的message对象,该message对象可以被Handler类的handleMessage(Message) 方法所处理。
post之类函数和sendmessage之类的函数都可以指定消息的执行时机,是立即执行、稍后一段时间执行,还是在某个确定时刻执行。这可以用来实现超时、消息或其他时间相关的操作。
当一个进程启动时,主线程独立执行一个消息队列,该队列管理着应用顶层的对象(如:activities、broadcast receivers等等)和所有创建的窗口。你可以创建自己的一个线程,并通过handler来与主线程进行通信。这可以通过在新的线程中调用主线程的handler的post和sendmessage操作来实现。
Handler类主要可以使用如下3个方法来设置执行Runnable对象的时间:
//立即执行Runnable对象publicfinalbooleanpost(Runnable r);//在指定时间uptimeMillis 后执行Runnable对象publicfinalbooleanpostAtTime(Runnable r,longuptimeMillis);//在指定的时间间隔delayMillis 执行Runnable对象publicfinalbooleanpostDelayed(Runnable r,longdelayMillis);
从上面的三个方法可以看出,第一个参数的类型都是Runnable ,因此,在调用这三个方法之前,需要有一个实现Runnable接口的类,Runnable接口的代码如下:
public interface Runnable
{
public voic run();//线程要执行的方法
}
在Runnable接口中只有一个Run 方法,该方法为线程执行方法。
Handlerhandler=newHandler();handler.postDelayed(This,5000);
如果想在5秒内停止记时可以用如下代码:
handler.removeCallbacks(this);
除此之外还可以用postAtTime方法指定未来的某一个精确时间来执行Runnable对象,代码如下:
Handler handler=new Handler();
handler.postAtTime(new RunToast(this){ },android.os.SystemClock.uptimeMillis()+15*1000);//在15秒后执行Runnable对象
其中RunToast是一个实现Runnable的接口的类,代码如下:
class RunToast implements Runnable
{
private Context context;
public RunaToast(Context context)
{
this.context;
}
@Override
public void run()
{
Toast.makeText(context,"15秒后显示Toast提示信息",Toast.LENGTH_LONG).show();
}
}
postAtTime的第二个参数表示一个精确的时间的毫秒数,如果从当前时间算起,需要使用android.os.SystemClock.uptimeMillis()获得基准时间。
要注意的是,不管使用哪个方法来执行Runnable对,都只能运行一次。如果想循环执行,必须在执行完后再次调用post、postAtTime或postDelayed方法。
例如,在Run方法中再次调用postDelayed方法,代码如下:
public void run()
{
tvCount.setText("Count:" + String.valueOf(++count));
//现次调用postDelayed方法,5秒后run方法仍被调用,然后再一次同用postDelayed方法,这样就行成了循环调用
handler.postDelayed(this, 5000);
}
下面是一个运用实例:
完整代码如下:
package sjl.handler;
import android.app.Activity;
import android.content.Context;
import android.os.Bundle;
import android.os.Handler;import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;
import android.widget.Toast;
public classcsfextendsActivityimplementsOnClickListener,Runnable{
private Handler handler;
private TextView tvCount;
private int count = 0;
classRunToastimplementsRunnable{
private Context context;
publicRunToast(Context context){
this.context = context;
}
@Override publicvoidrun(){
Toast.makeText(context, "15秒后显示Toast提示信息", Toast.LENGTH_LONG)
. show();
}
}
@Override publicvoidonClick(View view){
switch (view.getId())
{
case R.id.btnStart:
handler.postDelayed(this, 5000);
break;
case R.id.btnStop:
handler.removeCallbacks(this);
break;
case R.id.btnShowToast:
handler.postAtTime(new RunToast(this)
{
}, android.os.SystemClock.uptimeMillis() + 15 * 1000);
break;
}
}
@Override publicvoidrun(){
tvCount.setText("Count:" + String.valueOf(++count));
handler.postDelayed(this, 5000);
}
@Override publicvoidonCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
Button btnStart = (Button) findViewById(R.id.btnStart);
Button btnStop = (Button) findViewById(R.id.btnStop);
Button btnShowToast = (Button) findViewById(R.id.btnShowToast);
tvCount = (TextView) findViewById(R.id.tvCount);
btnStart.setOnClickListener(this);
btnStop.setOnClickListener(this);
btnShowToast.setOnClickListener(this);
handler = new Handler();
}
}
布局文件代码如下:main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="fill_parent"
android:layout_height="fill_parent">
<TextView
android:id="@+id/tvCount"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:textSize="20dp"/>
<Button
android:id="@+id/btnStart"
android:layout_width="fill_parent"
android:layout_height="wrap_content"android:text="开始计数"/>
<Button
android:id="@+id/btnStop"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="停止计数"/>
<Button
android:id="@+id/btnShowToast"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="15秒后显示Toast信息框"/>
</LinearLayout>
运行后,单击 [开始计数] 按钮,5秒后,会在按钮上方显示计数信息。然后单击 [15秒后显示Toast信息框] 按钮,过15秒后,会显示一个Toast信息框。