Android Handler 事件
Handler 的定义
Handler 主要接受子线程发送的数据, 并用此数据配合主线程更新 UI。 当应用程序启动时,Android 首先会开启一个主线程(也就是 UI 线程) ,主线程为管理界面中的 UI 控件,进行事件分发, 比如说,你要是点击一个 Button,Android 会分发事件到 Button 上,来响应你的操作。如果此时需要一个耗时的操作,例如: 联网读取数据,或者读取本地较大的一个文件的时候,你不能把这些操作放在主线程中,如果你放在主线程中的话,界面会出现假死现象, 如果 5 秒钟还没有完成的话,会收到 Android 系统的一个错误提示 强制关闭。
这个时候我们需要把这些耗时的操作,放在一个子线程中,因为子线程涉及到 UI 更新,Android 主线程不是线程安全的,也就是说,更新 UI 只能在主线程中更新,子线程中操作是危险的。这个时候,Handler 就出现了,来解决这个复杂的问题。Handler 运行在主线程中(UI 线程中),它与子线程可以通过 Message 对象来传递数据,这个时候,Handler 就承担着接受子线程传过来的(子线程用 sedMessage()
方法传递)Message 对象(里面包含数据),把这些消息放入主线程队列中,配合主线程进行更新 UI。
Handler 的一些特点
Handler 为 Android 操作系统中的线程通信工具,包为 android.os.Handler
。
与 Handler 绑定的有两个队列,一个为消息队列,另一个为线程队列。Handler 可以通过这两个队列来分别:
- 发送、接受、处理消息–消息队列;
- 启动、结束、休眠线程–线程队列;
Android OS 中,一个进程被创建之后,主线程(可理解为当前 Activity) 创建一个消息队列,这个消息队列维护所有顶层应用对象(Activities, Broadcast receivers 等) 以及主线程创建的窗口。你可以在主线程中创建新的线程,这些新的线程都通过 Handler 与主线程进行通信。通信通过新线程调用 Handler 的 post() 方法和 sendMessage() 方法实现,分别对应功能:
- post() 将一个线程加入线程队列;
- sendMessage() 发送一个消息对象到消息队列;
当然,post() 方法还有一些变体,比如
post(Runnable) postAtTime(Runnable,long) postDelayed(Runnable long) sendEmptyMessage(int) sendMessage(Message) sendMessageAtTime(Message,long) sendMessageDelayed(Message,long)
Handler 实例
public class MyHandlerActivity extends Activity {
Button button;
MyHandler myHandler;
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.handlertest);
button = (Button) findViewById(R.id.button);
myHandler = new MyHandler();
// 当创建一个新的 Handler 实例时, 它会绑定到当前线程和消息的队列中,开始分发数据
// Handler 有两个作用, (1) : 定时执行 Message 和 Runnalbe 对象
// (2): 让一个动作,在不同的线程中执行.
// 它安排消息,用以下方法
// post(Runnable)
// postAtTime(Runnable,long)
// postDelayed(Runnable,long)
// sendEmptyMessage(int)
// sendMessage(Message);
// sendMessageAtTime(Message,long)
// sendMessageDelayed(Message,long)
// 以上方法以 post 开头的允许你处理 Runnable 对象
//sendMessage() 允许你处理 Message 对象(Message 里可以包含数据,)
MyThread m = new MyThread();
new Thread(m).start();
}
/**
* 接受消息,处理消息 ,此 Handler 会与当前主线程一块运行
* */
class MyHandler extends Handler {
public MyHandler() {
}
public MyHandler(Looper L) {
super(L);
}
// 子类必须重写此方法,接受数据
@Override
public void handleMessage(Message msg) {
// TODO Auto-generated method stub
Log.d("MyHandler", "handleMessage......");
super.handleMessage(msg);
// 此处可以更新 UI
Bundle b = msg.getData();
String color = b.getString("color");
MyHandlerActivity.this.button.append(color);
}
}
class MyThread implements Runnable {
public void run() {
try {
Thread.sleep(10000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
Log.d("thread.......", "mThread........");
Message msg = new Message();
Bundle b = new Bundle();// 存放数据
b.putString("color", "我的");
msg.setData(b);
MyHandlerActivity.this.myHandler.sendMessage(msg); // 向 Handler 发送消息,更新 UI
}
}
}
简单的说,Activity 的 onCreate 方法里启动一个线程,在这个线程的 run 方法里使用一个 Message 对象并使用 Handler 的 sendMessage 方法发送到队列中,最后在 Activity 里 new 一个 Handler 的内部类实现 handMessage 方法,使用这个方法把队列中的 Message 对象取出来以实现异步操作。
然后是 post 的例子,这里稍微说一下,直接使用 new Handler().post(Runnable)
这样的写法并没有新开线程,也就是说依然是在主线程中执行,相当于简单调用了线程的 run 方法而不是 start 方法。这个有人说是 android 的 bug,解决方案是这样使用:
HandlerThread handlerThread = new HandlerThread("myHandlerThread");
handlerThread.start();
handler = new Handler(handlerThread.getLooper());
来看一个完整的 post 例子:
public class MyThread extends Activity {
private Handler handler = null;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
HandlerThread handlerThread = new HandlerThread("myHandlerThread");
handlerThread.start();
handler = new Handler(handlerThread.getLooper());
handler.post(new MyRunnable());
System.out.println("Oncreate---The Thread id is :"
+ Thread.currentThread().getId());
setContentView(R.layout.main);
}
private class MyRunnable implements Runnable {
public void run() {
System.out.println("Runnable---The Thread is running");
System.out.println("Runnable---The Thread id is :"
+ Thread.currentThread().getId());
try {
Thread.sleep(6000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
在这个 demo 中,用到了 HandlerThread,在 HandlerThread 对象中可以通过 getLooper 方法获取一个 Looper 对象控制句柄,我们可以将其这个 Looper 对象映射到一个 Handler 中去来实现一个线程同步机制。于是就有以下结果;
1:控制台的输出:
Oncreate---The Thread id is :1
Runnable---The Thread is running
Runnable---The Thread id is :10
2:程序启动后,我们立刻看到 main.xml 中的内容。这样就达到了多线程的结果。
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
上一篇: Rails Cache 缓存
下一篇: 彻底找到 Tomcat 启动速度慢的元凶
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论