为了更好的理解Handler的工作原理,先介绍一下与Handler一起工作的几个组件:
- Message:Handler接受和处理的消息对象
- Looper:每个线程只能拥有一个Looper。它的loop方法负责读取MessageQueue中的消息,读到信息之后就把消息交给发送该消息的Handler处理
- MessageQueue:消息队列,它采用先进先出的方式来管理Message。程序创建Looper对象时,会在它的构造器中创建MessageQueue对象
Looper、MessageQueue、Handler各自作用如下:
- Looper:每个线程只有一个Looper,它负责管理MessageQueue,会不断的从MessageQueue中取出消息,并将消息分给对应的Handler处理
- MessageQueue:由Looper负责管理。它采用先进先出的方式管理Message
- Handler:它能把消息发送给Looper管理的MessageQueue,并负责处理Looper分给它的消息
在线程中使用Handler的步骤如下:
- 调用Loop的prepare()方法为当前线程创建Looper对象,创建Looper对象时,它的构造器会创建与之配套的MessageQueue
- 有了Looper后,创建Handler子类的实例,重写handleMessage实例()方法,该方法负责处理来自于其他线程的消息
- 调用Looper的loop()方法启动Looper
实例:使用新线程计算质数
MainActivity.java
import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
import android.os.Message;
import android.os.PersistableBundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.Toast;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import java.util.ArrayList;
import java.util.List;
public class MainActivity extends Activity {
static final String UPPER_NUM = "upper";
EditText etNum;
CalThread calThread;
class CalThread extends Thread {
public Handler mHandler;
public void run() {
Looper.prepare();
mHandler = new Handler() {
@Override
public void handleMessage(@NonNull Message msg) {
//super.handleMessage(msg);
if (msg.what == 0x123) {
int upper = msg.getData().getInt(UPPER_NUM);
List<Integer> nums = new ArrayList<Integer>();
//计算从2开始,到upper的所有质数
outer:
for (int i = 2; i <= upper; i++) {
for (int j = 2; i <= Math.sqrt(i); j++) {
if (i != 2 && i % j == 0) {
continue outer;
}
}
nums.add(i);
}
Toast.makeText(MainActivity.this, nums.toString(), Toast.LENGTH_LONG).show();
}
}
};
Looper.loop();
}
}
@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
etNum = findViewById(R.id.etNum);
calThread = new CalThread();
calThread.start();
}
public void cal(View v) {
Message message = new Message();
message.what = 0x123;
Bundle bundle = new Bundle();
bundle.putInt(UPPER_NUM, Integer.parseInt(etNum.getText().toString()));
message.setData(bundle);
calThread.mHandler.sendMessage(message);
}
}
main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<EditText
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/etNum"/>
<Button
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:onClick="cal"/>
</LinearLayout>