9.1   服务是什么

服务(Service)是 Android 中实现程序后台运行的解决方案,它非常适合用于去执行那 些不需要和用户交互而且还要求长期运行的任务。服务的运行不依赖于任何用户界面,即使 当程序被切换到后台,或者用户打开了另外一个应用程序,服务仍然能够保持正常运行。

不过需要注意的是,服务并不是运行在一个独立的进程当中的,而是依赖于创建服务 时所在的应用程序进程。当某个应用程序进程被杀掉时,所有依赖于该进程的服务也会停 止运行。

另外,也不要被服务的后台概念所迷惑,实际上服务并不会自动开启线程,所有的代码 都是默认运行在主线程当中的。也就是说,我们需要在服务的内部手动创建子线程,并在这 里执行具体的任务,否则就有可能出现主线程被阻塞住的情况。那么本章的第一堂课,我们 就先来学习一下关于 Android 多线程编程的知识。

9.2   Android 多线程编程

熟悉 Java 的你,对多线程编程一定不会陌生吧。当我们需要执行一些耗时操作,比如说发起一条网络请求时,考虑到网速等其他原因,服务器未必会立刻响应我们的请求,如果

不将这类操作放在子线程里去运行,就会导致主线程被阻塞住,从而影响用户对软件的正常 使用。那么就让我们从线程的基本用法开始学习吧。

9.2.1    线程的基本用法

Android 多线程编程其实并不比 Java 多线程编程特珠,基本都是使用相同的语法。比如 说,定义一个线程只需要新建一个类继承自 Thread,然后重写父类的 run()方法,并在里面 编写耗时逻辑即可,如下所示:

class MyThread extends Thread {

@Override

public void run() {

// 处理具体的逻辑

}

}

那么该如何启动这个线程呢?其实也很简单,只需要 new 出 MyThread 的实例,然后调 用它的 start()方法,这样 run()方法中的代码就会在子线程当中运行了,如下所示:

new MyThread().start();

当然,使用继承的方式耦合性有点高,更多的时候我们都会选择使用实现 Runnable 接 口的方式来定义一个线程,如下所示:

class MyThread implements Runnable {

@Override

public void run() {

// 处理具体的逻辑

}

}

如果使用了这种写法,启动线程的方法也需要进行相应的改变,如下所示:

MyThread myThread = new MyThread();

new Thread(myThread).start();

可以看到,Thread 的构造函数接收一个 Runnable 参数,而我们 new 出的 MyThread 正是 一个实现了 Runnable 接口的对象,所以可以直接将它传入到 Thread 的构造函数里。接着调用 Thread 的 start()方法,run()方法中的代码就会在子线程当中运行了。当然,如果你不想专门再定义一个类去实现 Runnable 接口,也可以使用匿名类的方式, 这种写法更为常见,如下所示:

new Thread(new Runnable() {

@Override

public void run() {

// 处理具体的逻辑

}

}).start();

以上几种线程的使用方式相信你都不会感到陌生,因为在 Java 中创建和启动线程也是 使用同样的方式。了解了线程的基本用法后,下面我们来看一下 Android 多线程编程与 Java 多线程编程不同的地方。

9.2.2    在子线程中更新 UI

和许多其他的 GUI 库一样,Android 的 UI 也是线程不安全的。也就是说,如果想要更 新应用程序里的 UI 元素,则必须在主线程中进行,否则就会出现异常。眼见为实,让我们通过一个具体的例子来验证一下吧。新建一个 AndroidThreadTest 项 目,然后修改 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:id="@+id/change_text" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="Change Text" />

<TextView android:id="@+id/text" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_centerInParent="true" android:text="Hello world" android:textSize="20sp" />

</RelativeLayout>

布局文件中定义了两个控件,TextView 用于在屏幕的正中央显示一个 Hello world 字符 串,Button 用于改变 TextView 中显示的内容,我们希望在点击 Button 后可以把 TextView 中 显示的字符串改成 Nice to meet you。

接下来修改 MainActivity 中的代码,如下所示:

public class MainActivity extends Activity implements OnClickListener {

private TextView text;

private Button changeText;

@Override

protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main);

text = (TextView) findViewById(R.id.text);

changeText = (Button) findViewById(R.id.change_text);

changeText.setOnClickListener(this);

}

@Override

public void onClick(View v) {

switch (v.getId()) {

case R.id.change_text:

new Thread(new Runnable() {

@Override

public void run() {

text.setText("Nice to meet you");

}

}).start();

break;

default:

break;

}

}

}

可以看到,我们在 Change Text 按钮的点击事件里面开启了一个子线程,然后在子线程中调用 TextView 的 setText()方法将显示的字符串改成 Nice to meet you。代码的逻辑非常简 单,只不过我们是在子线程中更新 UI 的。现在运行一下程序,并点击 Change Text 按钮,你 会发现程序果然崩溃了,如图 9.1 所示。

图   9.1

然后观察 LogCat 中的错误日志,可以看出是由于在子线程中更新 UI 所导致的,如图 9.2所示。

图   9.2

由此证实了 Android 确实是不允许在子线程中进行 UI 操作的。但是有些时候,我们必 须在子线程里去执行一些耗时任务,然后根据任务的执行结果来更新相应的 UI 控件,这该 如何是好呢?

对于这种情况,Android 提供了一套异步消息处理机制,完美地解决了在子线程中进行 UI 操作的问题。本小节中我们先来学习一下异步消息处理的使用方法,下一小节中再去分 析它的原理。

修改 MainActivity 中的代码,如下所示:

public class MainActivity extends Activity implements OnClickListener {

public static final int UPDATE_TEXT = 1;

private TextView text;

private Button changeText;

private Handler handler = new Handler() {

public void handleMessage(Message msg) {

switch (msg.what) {

case UPDATE_TEXT:

// 在这里可以进行UI操作

text.setText("Nice to meet you");

break;

default:

break;

}

}

};

……

@Override

public void onClick(View v) {

switch (v.getId()) {

case R.id.change_text:

new Thread(new Runnable() {

@Override

public void run() {

Message message = new Message();

message.what = UPDATE_TEXT;

handler.sendMessage(message); //Message对象发送出去

}

}).start();

break;

default:

break;

}

}

}

这里我们先是定义了一个整型常量 UPDATE_TEXT,用于表示更新 TextView 这个动作。 然后新增一个 Handler 对象,并重写父类的 handleMessage 方法,在这里对具体的 Message 进行处理。如果发现 Message 的 what 字段的值等于 UPDATE_TEXT,就将 TextView 显示的 内容改成 Nice to meet you。

下面再来看一下 Change Text 按钮的点击事件中的代码。可以看到,这次我们并没有在 子线程里直接进行 UI 操作,而是创建了一个 Message(android.os.Message)对象,并将它 的 what 字段的值指定为 UPDATE_TEXT ,然后调用 Handler 的 sendMessage() 方法将这条 Message 发送出去。很快,Handler 就会收到这条 Message,并在 handleMessage()方法中对它 进行处理。注意此时 handleMessage()方法中的代码就是在主线程当中运行的了,所以我们可 以放心地在这里进行 UI 操作。接下来对 Message 携带的 what 字段的值进行判断,如果等于 UPDATE_TEXT,就将 TextView 显示的内容改成 Nice to meet you。

现在重新运行程序,可以看到屏幕的正中央显示着 Hello world。然后点击一下 ChangeText 按钮,显示的内容着就被替换成 Nice to meet you,如图 9.3 所示。

图   9.3

这样你就已经掌握了 Android 异步消息处理的基本用法,使用这种机制就可以出色地解决掉在子线程中更新 UI 的问题。不过恐怕你对它的工作原理还不是很清楚,下面我们就来 分析一下 Android 异步消息处理机制到底是如何工作的。

9.2.3   解析异步消息处理机制

Android 中的异步消息处理主要由四个部分组成,Message、Handler、MessageQueue 和 Looper。其中 Message 和 Handler 在上一小节中我们已经接触过了,而 MessageQueue 和 Looper 对于你来说还是全新的概念,下面我就对这四个部分进行一下简要的介绍。

1.    Message

Message 是在线程之间传递的消息,它可以在内部携带少量的信息,用于在不同线 程之间交换数据。上一小节中我们使用到了 Message 的 what 字段,除此之外还可以使 用 arg1 和 arg2 字段来携带一些整型数据,使用 obj 字段携带一个 Object 对象。

2.    Handler

Handler 顾名思义也就是处理者的意思,它主要是用于发送和处理消息的。发送消 息一般是使用 Handler 的 sendMessage()方法,而发出的消息经过一系列地辗转处理后, 最终会传递到 Handler 的 handleMessage()方法中。

3.    MessageQueue

MessageQueue 是消息队列的意思,它主要用于存放所有通过 Handler 发送的消息。 这部分消息会一直存在于消息队列中,等待被处理。每个线程中只会有一个 MessageQueue 对象。

4.    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 操作了。整个异步 消息处理机制的流程示意图如图 9.4 所示。

图   9.4

一条 Message 经过这样一个流程的辗转调用后,也就从子线程进入到了主线程,从不能 更新 UI 变成了可以更新 UI,整个异步消息处理的核心思想也就是如此。

android: 多线程编程基础相关推荐

  1. Android多线程编程(一)- 线程基础

    Android多线程编程(一) 到了学习下一部分了,多线程编程一直都是Android经常使用的部份,在进行网络请求或者访问数据库时,为了避免主线程被耗时操作阻塞了进度,通常都会开启子线程进行处理,多线 ...

  2. 线程与进程、Android多线程编程

    线程与进程.Android多线程编程 目录 一.概念 1.程序 2.进程 3.并发 4.并行 5.线程 二.Android中的进程与线程 1.安卓的进程 2.安卓的线程 三.Android多线程编程 ...

  3. Android开发实例——倒计时器——Android多线程编程

    文章目录 一.倒计时器功能描述 二.实现步骤 1.新建Activity,命名为ThreadDemo 2.设计ThreadDemo 的布局文件 3.编写ThreadDemo.java 文件代码 3.1 ...

  4. android 多线程编程

    主线程的职责是创建.显示和更新UI控件.处理UI事件.启动子线程.停止子线程;子线程的职责是进行处理和向主线程发出更新UI消息,而不是直接更新UI. android 多线程编程报错: 05-14 15 ...

  5. Android多线程编程

    Android多线程编程 其实就是将耗时操作放在子线程里运行,防止主线程被阻塞,影响软件使用. 线程的基本用法 定义一个线程只需要新建一个类继承自Thread,然后重写父类的run()方法,并在里面编 ...

  6. Android网络编程基础(一) - 基础知识

    内容介绍:Android网络编程基础 博客地址:http://blog.csdn.net/kevindgk 版权声明:本文为原创文章,未经允许不得转载 联系方式:815852777@qq.com 前言 ...

  7. Delphi多线程编程基础入门

    1. 概述 对于开发人员来说,多线程是必备的知识,但相对来说,也是比较难的知识点.Delphi是一门古老而优秀的编程语言,它对多线程的处理有一些特殊的地方,本文尝试做一些简单的讲解,可以当作Delph ...

  8. 多线程编程——基础语法篇

    多线程编程 文章目录 多线程编程 一.Thread 1.1 Thread用法一 1.2.Thread用法二 (Runnable) 1.3.Thread用法三 1.4.Thread用法四 1.5.Thr ...

  9. 高并发下Java多线程编程基础

    摘要: Java线程同步与异步 线程池 无锁化的实现方案 分布锁的实现方案 分享的目的: 进一步掌握多线程编程和应用的技巧,希望对大家在平时的开发中应对高并发编程有所帮助 Java线程同步与异步 1. ...

最新文章

  1. 修改IDEA项目的JDK应用路径
  2. Java 18 要来了,你不会还在用Java 8吧?
  3. xmarin.android导航栏,android – 如何在xamarin表单中更改导航页面后退按钮
  4. Responsive设计——meta标签
  5. 如何构建一个有效的知识库?
  6. 网站优化过度后会出现哪些“后遗症”?悠着点~
  7. 前端基础:技术栈简介
  8. 面向java开发者的函数式编程_函数式编程让你忘记设计模式
  9. java 找到一行 更换单词_Java实现对一行英文进行单词提取功能示例
  10. python转义引号的作用_在Jinja2中渲染时转义双引号
  11. 中控考勤机二次开发 java_SDK二次开发,读取中控考勤机打卡记录测试。
  12. Involution: Inverting the Inherence of Convolution for Visual Recognition(CVPR2021)
  13. C语言 空气质量优良率
  14. 伽卡他卡学生端卸载方案
  15. 开发实况4.1.linux相关-CRT连接虚拟机提示用户名或密码错误
  16. 2018年年初的面试经验谈
  17. 推荐几个面向 Web 开发者的杀手级网站
  18. 查看LINUX放开端口,linux如何查看端口是否开放?
  19. Spring AOP 学习笔记
  20. 演进式架构学习笔记(一):架构评估及适应度函数

热门文章

  1. java future 返回值_Java--Callable与返回值future
  2. sybase jz0c0 连接已关闭_Go 基于 channel 实现连接池
  3. c++ main函数调用 类中的枚举_为什么 Java 的 main 方法必须是 public static void?
  4. xp 无法用计算机名访问,WinXP工作组计算机无法访问的解决方法
  5. mysql与oracle存储过程_MySQL与Oracle差异比较之五存储过程Function
  6. html鼠标变成小手_什么牌子的鼠标好用?2020双十一鼠标选购建议和产品推荐
  7. 搜索不包含关键词_关键词排名:搜索同一个关键词,每次的排名却不一样?
  8. qmc0文件怎么转换mp3_音频转换器哪个好 怎么剪切MP3音频制作手机铃声
  9. mac 上mysql怎么卸载不了_mac上mysql怎么卸载不了
  10. python网络编程知识_python六十七课——网络编程(基础知识了解)