在这本书里首先介绍了怎么开启线程,方式和java是一样的。然后说明了在子线程中更新UI是不允许的,必须通过发送Message、用Handler接收来实现。
1、解析异步消息处理机制
由于整个过程比较复杂,所以我还是用第一行代码的原文进行解释:
①. Message
Message 是在线程之间传递的消息,它可以在内部携带少量的信息,用于在不同线程之间交换数据。上一小节中我们使用到了Message 的what 字段,除此之外还可以使用arg1 和arg2 字段来携带一些整型数据,使用obj 字段携带一个Object 对象。
②. Handler
Handler 顾名思义也就是处理者的意思,它主要是用于发送和处理消息的。发送消息一般是使用Handler 的sendMessage()方法,而发出的消息经过一系列地辗转处理后,最终会传递到Handler 的handleMessage()方法中。
③. MessageQueue
MessageQueue 是消息队列的意思,它主要用于存放所有通过Handler 发送的消息。这部分消息会一直存在于消息队列中,等待被处理。每个线程中只会有一个MessageQueue对象。
④. Looper
Looper 是每个线程中的MessageQueue 的管家,调用Looper 的loop()方法后,就会进入到一个无限循环当中,然后每当发现MessageQueue 中存在一条消息,就会将它取出,并传递到Handler 的handleMessage()方法中。每个线程中也只会有一个Looper 对象。
了解了Message、Handler、MessageQueue 以及Looper 的基本概念后,我们再来对异步消息处理的整个流程梳理一遍。首先需要在主线程当中创建一个Handler 对象,并重写handleMessage()方法。然后当子线程中需要进行UI 操作时,就创建一个Message 对象,并通过Handler 将这条消息发送出去。之后这条消息会被添加到MessageQueue 的队列中等待被处理,而Looper 则会一直尝试从MessageQueue 中取出待处理消息,最后分发回Handler的handleMessage()方法中。由于Handler 是在主线程中创建的,所以此时handleMessage()方法中的代码也会在主线程中运行,于是我们在这里就可以安心地进行UI 操作了。如下图所示:
进行handler测试的代码也比较简单,这里就不展开讲了
// AndroidThreadTest-MainActivity.java
private Button mBChangeText;
private TextView mTV;
private final String SET_TEXT = "文本已改变!";
private final int CHANGE_TEXT = 0x10;
private Handler handler = new Handler() {
@Override
public void handleMessage(Message msg) {
switch (msg.what) {
case CHANGE_TEXT:
String change_text = (String) msg.obj;
mTV.setText(change_text);
super.handleMessage(msg);
break;
default:
break;
}
}
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mBChangeText = (Button) findViewById(R.id.change_text_button);
mTV = (TextView) findViewById(R.id.textview);
mBChangeText.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View arg0) {
Message message = new Message();
message.what = CHANGE_TEXT;
message.obj = SET_TEXT;
handler.sendMessage(message);
}
});
}
<!-- AndroidThreadText-activity_main.xml -->
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<Button android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="改变文本内容"
android:id="@+id/change_text_button"/>
<TextView android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:text="@string/hello_world"
android:id="@+id/textview"
android:textSize="20sp"/>
</RelativeLayout>