9.1 服务是什么
服务是实现后台运行程序的解决方案,适合用于执行那些不需要和用户交互并长期运行的任务。
服务并不运行在一个独立的进程中,而是依赖于创建服务时所在的应用程序进程。当某个应用程序进程被杀掉时,所有依赖于该进程的服务也会停止运行。
服务并不会自动开启线程,所有代码都是默认运行在主线程中的,需要手动创建子线程,不然可能出现主线程阻塞的情况。
9.2 Android多线程编程
9.2.1 线程的基本用法
创建多线程的方式有两种,如下所示:
1. 继承Thread类,重写run()方法。
class MyThread extends Thread{
@Override
public void run() {
super.run();
//...
}
}
//启动线程
MyThread myThread=new MyThread();
myThread.start();
2.实现Runnable接口
class MyThread implements Runnable{
@Override
public void run() {
//...
}
}
//启动线程
new Thread(new MyThread()).start();
或者使用匿名内部类的方式:
//使用匿名内部类方式
new Thread(new Runnable() {
@Override
public void run() {
//...
}
}).start();
9.2.2 异步消息处理机制
(1)为什么要用异步消息处理机制?
因为
Android中的UI是线程不安全的,所以对UI的操作只能在主线程中进行,
Android中不允许在子线程中进行UI操作,但是有时候我们必须要在子线程中进行一些耗时操作,并根据操作结果来更新UI控件。
(2)解析异步消息处理机制
Android中异步消息机制主要由四个部分组成:
Message、
Handler、
MessageQueue和
Looper。
- Message:在线程之间传递的消息,可在内部携带少量信息,用于在不同线程之间交换数据。可以使用Message的what、arg1、arg2来携带整型数据,使用obj字段携带object对象。
- Handler:处理者,主要用于发送和处理消息的。发送消息一般使用Handler的sendMessage()方法(实例方法,下同),而发出的消息经过一系列辗转之后最终回到Handler的handleMessage()方法中。
- MessageQueue:消息队列,主要存放所有通过Handler发送的消息。每个线程只会有一个MessageQueue对象。
- Looper:MessageQueue的管家,通过Looper的 loop()方法,进入到无线循环,每当发现MessageQueue中存在一个消息,就会将它取出来传递给Handler的HandleMessage()方法。每个线程中只会有一个Looper对象。
(3)整体流程
首先在主线程中创建一个 Handler 对象,并重写handleMessage()方法做其他操作。
然后当子线程中需要进行 UI 操作时,就创建一个 Message 对象,并通过 Handler 将这条消息发送出去。
之后这条消息会被添加到MessageQueue 的队列中等待被处理,而 Looper 则会一直尝试从 MessageQueue 中取出待处理消息,最后分发回 Handler的 handleMessage()方法中。
由于 Handler是在主线程中创建的,所以此时 handleMessage()方法中的代码也会在主线程中运行,于是我们在这里就可以安心地进行 UI 操作了。整个流程如图所示:
(4)示例
实现点击活动上的按钮并改变下方TextView显示的文字内容。
public class MainActivity extends Activity implements View.OnClickListener {
//定义一个操作标识
public static final int UPDATE_TEXT = 1;
//控件
private Button changeText;
private TextView text;
/**
* 2.创建一个Handler对象,并重写handleMessage()方法
* 在方法中对UI进行其他操作
*/
private Handler handler = new Handler() {
@Override
public void handleMessage(Message msg) {
switch (msg.what) {
case UPDATE_TEXT:
text.setText("我变了我变了~");
break;
default:
break;
}
}
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
/**
* 1.获取控件对象,