Android 中Thread,Handler,Loop学习

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中打印空间内容就可验证。

时间: 2024-10-12 22:44:42

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

Android中Thread、Handler、Looper、MessageQueue的原理分析

在Android开发当中,Thread.Handler.Looper这几个类是特别常见,在刚开始学习Android的时候对这些类可能并不是很清晰.下面我们就一起从源码的角度剖析一下这几个类的工作原理. Thread 首先是Thread, 我们都知道一个Thread就是一个线程对象,只要在run方法中填写自己的代码然后启动该线程就可以实现多线程操作.例如 : new Thread(){ public void run() { // 耗时的操作 }; }.start(); 我们知道,针对上面的代码中

Android中的Handler, Looper, MessageQueue和Thread

Android中的Handler, Looper, MessageQueue和Thread 前几天,和同事探讨了一下Android中的消息机制,探究了消息的发送和接收过程以及与线程之间的关系.虽然我们经常使用这些基础的东西,但对于其内部原理的了解,能使我们更加容易.合理地架构系统,并避免一些低级错误. 对于这部分的内容,将分成4小节来描述: 1.职责与关系 2.消息循环 3.线程与更新 4.几点小结 ------------------------------------------------

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

本文主要是对Handler和消息循环的实现原理进行源码分析,如果不熟悉Handler可以参见博文< Android中Handler的使用>,里面对Android为何以引入Handler机制以及如何使用Handler做了讲解. 概括来说,Handler是Android中引入的一种让开发者参与处理线程中消息循环的机制.我们在使用Handler的时候与Message打交道最多,Message是Hanlder机制向开发人员暴露出来的相关类,可以通过Message类完成大部分操作Handler的功能.但

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

在第二篇文章<Android中利用Handler实现消息的分发机制(一)>中,我们讲到主线程的Looper是Android系统在启动App的时候,已经帮我们创建好了,而如果在子线程中需要去使用Handler的时候,我们就需要显式地去调用Looper的 prepare方法和loop方法,从而为子线程创建其唯一的Looper. 具体代码如下: class LooperThread extends Thread { public Handler mHandler; public void run()

Android中的Handler

当我们在处理下载或是其他需要长时间执行的任务时,如果直接把处理函数放在Activity的OnCreate或是OnStart中,会导致执行过程中整个Activity无响应,如果时间过长,程序就会挂掉.Handler就是把这些功能放到一个当初的线程里执行,与Activity互不影响. 一.Handler的定义:  Handler主要接收子线程发送的数据, 并用此数据配合主线程更新UI,用来跟UI主线程交互用.比如可以用handler发送一个message,然后在handler的线程中来接收.处理该消

android中的Handler和Runnable

最近在做一个项目,在网络请求时考虑用Handler进行处理,然后就研究了一下Handler和Runnable 首先在看一下java中的Runnable The Runnable interface should be implemented by any class whose instances are intended to be executed by a thread. The class must define a method of no arguments called run.

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

概述: 很多android初学者对android 中的handler不是很明白,其实Google参考了Windows的消息处理机制, 在Android系统中实现了一套类似的消息处理机制.在下面介绍handler机制前,首先得了解以下几个概念:     1. Message 消息,理解为线程间通讯的数据单元.例如后台线程在处理数据完毕后需要更新UI,则可发送一条包含更新信息的Message给UI线程.     2. Message Queue 消息队列,用来存放通过Handler发布的消息,按照先

Android中的Handler的具体用法

Android UI操作并不是线程安全的并且这些操作必须在UI线程中执行.Android利用Handler来实现UI线程的更新的. Handler是Android中的消息发送器,其在哪个Activity中创建就属于且紧紧属于该Activity.还可以说其在哪个线程中new的,就是那个线程的Handler. Handler的定义: 主要接受子线程发送的数据, 并用此数据配合主线程更新UI. 解释: 当应用程序启动时,Android首先会开启一个主线程 (也就是UI线程) , 主线程为管理界面中的U

Android中关于JNI 的学习(五)在C文件中使用LogCat

事实上,本文是在Peter Jerde的How much information can be stored by ordering 52 playing cards文章基础上翻译.改编和扩展而来的.当然这是经过Jerde本人首肯的. 注意本文方法并非最优,也没有完全利用所有的信息空间,只是简单的尝试. 有数字的地方就有信息.所以扑克牌中保存信息不是什么新鲜事. PDF文档点这里:下载 原文(英文)点这里:访问 这里有两个DEMO. 编码DEMO,解码DEMO 首先是"DEEP IN SHALL