1.先看一下最简单的进度条示例

EG:

package com.sxz.android.thread;

import java.util.concurrent.atomic.AtomicBoolean;
import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.widget.ProgressBar;
public class HandlerDemo extends Activity {

ProgressBar bar;
    AtomicBoolean isRunning = new AtomicBoolean(false);
    Handler mHandler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            bar.incrementProgressBy(5);
        }
    };

@Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.my_progressbar);
        bar = (ProgressBar) findViewById(R.id.progressBar1);
    }

@Override
    protected void onStart() {
        super.onStart();
        Thread myThread = new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    for (int i = 0; i < 20 && isRunning.get(); i++) {
                        Message msg = mHandler.obtainMessage();
                        Thread.sleep(500);
                        msg.sendToTarget();
                    }

} catch (Throwable t)
                {
                }
            }
        });

isRunning.set(true);
        myThread.start();
    }

@Override
    protected void onStop() {
        super.onStop();
        isRunning.set(false);
    }
}

2. 基本概念

MessageQueue 消息队列,存放消息的地方,按照FIFO规则执行,每一个线程只可以拥有一个MessageQueue,在创建 Looper 对象会创建一个MessageQueue对象。而MessageQueue

都会有一个对应的 Handler,Handler会向 MessageQueue通过两种方法发送消息.

① sendMessage. 通过 sendMessage发送的是一个 message 对象,会被 Handler的 handleMessage() 函数处理。
② post 通过 post 方法发送的是一个 runnable 对象,则会自己执行。

这两种消息都会插在 MessageQueue 队尾并按先进先出的方式执行.但是通过这两种方法发送出去的消息执行的方式略有不同.

Message 消息对象,MessageQueue中存放的对象。一个 MessageQueue中可以包含多个Message 对象。可以通过 Message.obtain() 或者 Handler.obtainMessage() 获取 Message 对象.但是这并不一定是直接创建一个新的实例.而是先从消息池中看有没有可用的 Message实例。存在则直接取出后返回这个实例。如果消息池中没有可用的 Message 实例,则用给定的参数创建一个Message 对象。调用 removeMessage()时,将 Message 从 MessageQueue 中删除,同时放入到消息池中。
消息,其实可以理解为线程间交流的信息,处理数据后台线程需要更新 UI,则发送Message内含一些数据给 UI线程.

Looper 操作 MessageQueue,一个 Looper 对应一个 MessageQueue.通过调用 Looper.myLooper() 可以获取当前线程的 Looper对象.Looper 从 MessageQueue中取出 Message然后, 交由Handler的 handleMessage() 进行处理。处理完成后,调用 Message.recycle()将其放入消息池中.

Looper 是每条线程里的 MessageQueue的管家。Android没有 Gloabal 的 MessageQueue,而 Android 会自动替主线程(UI线程) 建立MessageQueue,但在子线程里并没有建立MessageQueue。所以调用 Looper.getMainLooper() 得到的主线程Looper不为 NULL,但是调用Looper.myLooper()得到的Looper就有可能为 NULL。

总结:几者之间的关系:

Handler 消息的处理者。

handler 负责将需要传递的信息封装成 Message 对象,

然后调用 sendMessage() 方法将消息放入 MessageQueue中,

当MessageQueue循环到该 Message,

调用相应的handler 对象的 handleMessage() 方法对其进行处理。

Handler 都可以共享同一个 Looper 和 MessageQueue.

3. handler.post(r)同一个线程的疑惑

handler.post(r);是把r加到消息队列,但并未开辟新线程。等到消息被取出时才执行。

package com.lei.handlethread;

import android.os.Bundle;
import android.os.Handler;
import android.app.Activity;
import android.view.Menu;
import android.widget.Button; public class MainActivity extends Activity {
private Button btn = null;
private Handler handler = new Handler();
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
handler.post(r);
setContentView(R.layout.activity_main);
btn = (Button)findViewById(R.id.hello);// 用来验证setContentView()先执行的。
String s=(String) btn.getText();//
System.out.println(s);
System.out.println("activity--->"+Thread.currentThread().getId());
System.out.println("Activityname--->"+Thread.currentThread().getName());
} Runnable r = new Runnable() {
public void run() {
System.out.println("handler--->"+Thread.currentThread().getId());
System.out.println("handlername--->"+Thread.currentThread().getName());
try {
Thread.sleep(10000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
} }; @Override
public boolean onCreateOptionsMenu(Menu menu) {
getMenuInflater().inflate(R.menu.activity_main, menu);
return true;
}
}

运行结果:logCat先打印如下信息。程序运行界面过10s显示TextView文字。

解释:

main线程从消息泵中取出一个消息,处理(执行相关函数),然后再取一个,处理。所以onCreate是某一消息处理中的执行,其中post一个消息,只是把消息加入队列了,还没执行新消息,什么时候执行?要等前一个消息处理完,再次从消息泵中取消息处理时,它才被执行。所以先是main的system.out,再是post的system.out

相比之下,sendMessage是同步执行,用handler.sendMessage,那顺序就变了。

至于setContentView(R.layout.activity_main);肯定是最先执行,程序界面最先打开了,但是界面空间要等到Activity的Resume(即交互阶段)阶段才会显示。

通过获取界面空间ID,在Log中打印空间内容就可验证。

Android 中Thread,Handler,Loop学习的更多相关文章

  1. Android中利用Handler实现消息的分发机制(三)

    在第二篇文章<Android中利用Handler实现消息的分发机制(一)>中,我们讲到主线程的Looper是Android系统在启动App的时候,已经帮我们创建好了,而假设在子线程中须要去 ...

  2. 深入源代码解析Android中的Handler,Message,MessageQueue,Looper

    本文主要是对Handler和消息循环的实现原理进行源代码分析.假设不熟悉Handler能够參见博文< Android中Handler的使用>,里面对Android为何以引入Handler机 ...

  3. Android中的Handler的机制与用法详解

    概述: 很多android初学者对android 中的handler不是很明白,其实Google参考了Windows的消息处理机制, 在Android系统中实现了一套类似的消息处理机制.在下面介绍ha ...

  4. android中的Handler和Runnable

    最近在做一个项目,在网络请求时考虑用Handler进行处理,然后就研究了一下Handler和Runnable 首先在看一下java中的Runnable The Runnable interface s ...

  5. 转:Android中的Handler的机制与用法详解

    注:Message类的用法: message的几个参数都可以携带数据,其中arg1与arg2可以携带int类型,what是用户自定义的int型,这样接受者可以了解这个消息的信息. 说明:使用Messa ...

  6. Android中使用Handler造成内存泄露的分析和解决

    什么是内存泄露?Java使用有向图机制,通过GC自动检查内存中的对象(什么时候检查由虚拟机决定),如果GC发现一个或一组对象为不可到达状态,则将该对象从内存中回收.也就是说,一个对象不被任何引用所指向 ...

  7. Android中使用Handler造成内存泄露

    1.什么是内存泄露? Java使用有向图机制,通过GC自动检查内存中的对象(什么时候检查由虚拟机决定),如果GC发现一个或一组对象为不可到达状态,则将该对象从内存中回收.也就是说,一个对象不被任何引用 ...

  8. Android中的SQLite使用学习

    Android中的SQLite使用学习 SQLite是非常流行的嵌入式关系型数据库,轻载, 速度快,而且是开源.在Android中,runtime提供SQLite,所以我们可以使用SQLite,而且是 ...

  9. Android中关于Handler的若干思考

    在之前的博文中,讲过一些和Handler有关的知识,例如: Android 多线程----AsyncTask异步任务详解 Android多线程----异步消息处理机制之Handler详解 今天再把Ha ...

随机推荐

  1. 读书笔记 1 of Statistics :Moments and Moment Generating Functions (c.f. Statistical Inference by George Casella and Roger L. Berger)

    Part 1: Moments Definition 1 For each integer $n$, the nth moment of $X$, $\mu_n^{'}$ is \[\mu_{n}^{ ...

  2. 【C++/Qt】Qt中的parent形参

    在 派生类的构造函数初始化列表中 调用 父类的带有参数的构造函数,是为了初始化从父类继承来的成员变量.因为这些变量无法直接初始化,只能采用这种方式初始化. 而在qt中,MainWindow中的某成员变 ...

  3. mysql show status

    在LAMP架构的网站开发过程中,有些时候我们需要了解MySQL的服务器状态信息,譬如当前MySQL启动后的运行时间,当前MySQL的客户端会话连接数,当前MySQL服务器执行的慢查询数,当前MySQL ...

  4. WSDL2ObjC Unsupported Media Type

    调用WCF服务时,出这样的异常“415 Unsupported Media Type”, Because the WCF soap is v1.1, the http header should be ...

  5. Android 自动化测试 Emmagee

    Emmagee 是一个性能测试小工具 用来监控指定被测应用在使用过程中占用机器的CPU, 内存,流量资源的性能小工具 阅读目录 Emmagee 介绍 Emmagee是网易杭州研究院QA团队开发的一个简 ...

  6. Linux休眠,挂起,待机,关机的区别及相关命令

    休眠是一种更加省电的模式,它将内存中的数据保存于硬盘中,所有设备都停止工作.当再次使用时需按开关机键,机器将会恢复到您的执行休眠时的状态,而不用再次执行启动操作系统复杂的过程. 待机(挂起)是将当前处 ...

  7. consul模板配置参数值示例

    参看https://github.com/hashicorp/consul-template#examples // This is the address of the Consul agent. ...

  8. 《BI那点儿事》数据流转换——条件性拆分

    根据条件分割数据是一个在数据流中添加复杂逻辑的方法,它允许根据条件将数据输出到其他不同的路径中.例如,可以将TotalSugar< 27.4406的输出到一个路径,TotalSugar > ...

  9. CoreText原理及基本使用方法

    关于富文本的排版也是现在的一个技术点,以下是近日关于CoreText的学习记录以及个人理解,希望能对正在学习CoreText的朋友起到帮助. 1.框架坐标系 首先让我们先来看看CoreText坐标系和 ...

  10. cygwin编译环境小记

    [gcc] 预定义宏 编译器通常会根据平台和编译选项的不同,为被编译的程序提供不同的预定义的宏, 例如WIN32 WIN64 LINUX, 例如DEBUG, RELEASE. 1. 在使用gcc/g+ ...