- handler和handler的运行机制
- 为什么要用handler
- Looper
- MessageQueue
- handler怎么用
- 案例:倒计时
Handler是Android SDK中处理异步消息的核心类,让子线程通过与UI通信来刷新UI。子线程完成工作后通过handler发消息到massage queue,然后让looper从message queue传送消息给主线程。
handler通过接口和继承来实现
运行机制
1.创建一个handler对象,系统就把handler对象,UI线程和UI线程的消息队列捆绑起来。
2.当我们在线程中处理完数据后,可以通过handler对象将消息发出去。
3.消息会按先后顺序添加到消息队列中。
4.UI线程中的looper不断从消息队列中取出消息,刷新UI。
Handler
1.当应用程序启动时,会初始化一个UI线程。
2.UI线程中创建建了Looper,所以是一个循环工作线程。
3.创建Looper时,Looper会创建一个Message Queue。
4.UI线程中的Looper会不断从Message Queue中取出消息。
Looper
1.在Android中创建出的普通线程默认是没有消息循环的,run方法执行完毕,线程也就结束了。
2.如果让线程不停的循环工作时,可以使用Looper,将普通线程变成循环线程。
Message Queue
1.当创建Looper时,系统会自动弹出Message Queue。
3.当MessageQueue中有消息时,Looper讲从Message Queue中取出消息。
为什么要用handler:因为子线程不能刷新UI,主线程才能,但是耗时操作又不能放在主线程中,所以这个时候就要用handler来处理
代码如下,
package com.example.a17708.myapplication;
import android.os.Handler;
import android.os.Message;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
public class DownloadActivity extends AppCompatActivity implements View.OnClickListener {
private TextView downloadTV;
private Button downloadBtn;
private Handler handler = new Handler() {
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
downloadTV.setText("下载完成");
}
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_download);
bind();
}
private void bind() {
downloadBtn=findViewById(R.id.download_btn);
downloadTV=findViewById(R.id.download_tv);
downloadBtn.setOnClickListener(this);
}
@Override
public void onClick(View v) {
switch (v.getId()){
case R.id.download_btn:
new Thread(new Runnable() {
@Override
public void run() {
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
handler.sendEmptyMessage(1);
}
}).start();
break;
}
}
}
倒计时代码:
package com.example.a17708.myapplication;
import android.os.Handler;
import android.os.Message;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;
public class DownloadActivity extends AppCompatActivity implements View.OnClickListener {
private EditText daotime;
private TextView downloadTV,start_time;
private Button downloadBtn;
private int time;
private Handler handler = new Handler() {
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
if(msg.what==-1) {
start_time.setText("停止");
}else{
start_time.setText(msg.what+"秒");
}
}
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_download);
bind();
}
private void bind() {
daotime=findViewById(R.id.daotime);
start_time=findViewById(R.id.start_time);
downloadBtn=findViewById(R.id.download_btn);
downloadTV=findViewById(R.id.download_tv);
downloadBtn.setOnClickListener(this);
downloadTV.setOnClickListener(this);
}
@Override
public void onClick(View v) {
switch (v.getId()){
case R.id.download_btn:
start_time.setText(daotime.getText());
time=Integer.parseInt(daotime.getText().toString());
new Thread(new Runnable() {
@Override
public void run() {
try {
while (time>0){
handler.sendEmptyMessage(time);
time--;
Thread.sleep(5000);
}
} catch (InterruptedException e) {
e.printStackTrace();
}
handler.sendEmptyMessage(-1);
}
}).start();
break;
}
}
}
布局文件
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context=".DownloadActivity">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="50dp"
android:orientation="horizontal">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="30sp"
android:text="倒数计时"
android:id="@+id/download_tv"
android:layout_gravity="center"/>
<EditText
android:layout_width="100dp"
android:layout_height="50dp"
android:textSize="25sp"
android:id="@+id/daotime"/>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:orientation="horizontal"
android:gravity="center_horizontal|center_vertical"
android:layout_height="200dp">
<TextView
android:layout_width="50dp"
android:layout_height="50dp"
android:gravity="center"
android:textSize="30sp"
android:id="@+id/start_time"/>
</LinearLayout>
<Button
android:text="开始倒计时"
android:textSize="25sp"
android:id="@+id/download_btn"
android:layout_width="match_parent"
android:layout_height="50dp" />
</LinearLayout>
结果截图