【转】Android -- Looper.prepare()和Looper.loop()

Looper.prepare()和Looper.loop()

原文地址:http://blog.csdn.net/heng615975867/article/details/9194219

Android中的Looper类,是用来封装消息循环和消息队列的一个类,用于在android线程中进行消息处理。handler其实可以看做是一个工具类,用来向消息队列中插入消息的。

(1) Looper类用来为一个线程开启一个消息循环。    
默认情况下android中新诞生的线程是没有开启消息循环的。(主线程除外,主线程系统会自动为其创建Looper对象,开启消息循环。)    
Looper对象通过MessageQueue来存放消息和事件。一个线程只能有一个Looper,对应一个MessageQueue。

(2)

通常是通过Handler对象来与Looper进行交互的。Handler可看做是Looper的一个接口,用来向指定的Looper发送消息及定义处理
方法。    
默认情况下Handler会与其被定义时所在线程的Looper绑定,比如,Handler在主线程中定义,那么它是与主线程的Looper绑定。
mainHandler = new Handler() 等价于new Handler(Looper.myLooper()).
Looper.myLooper():获取当前进程的looper对象,类似的 Looper.getMainLooper()
用于获取主线程的Looper对象。

(3)
在非主线程中直接new Handler() 会报如下的错误: E/AndroidRuntime( 6173): Uncaught
handler: thread Thread-8 exiting due to uncaught exception
E/AndroidRuntime(
6173): java.lang.RuntimeException: Can‘t create handler inside thread
that has not called Looper.prepare()
原因是非主线程中默认没有创建Looper对象,需要先调用Looper.prepare()启用Looper。

(4) Looper.loop(); 让Looper开始工作,从消息队列里取消息,处理消息。

注意:写在Looper.loop()之后的代码不会被执行,这个函数内部应该是一个循环,当调用mHandler.getLooper().quit()后,loop才会中止,其后的代码才能得以运行。
(5) 基于以上知识,可实现主线程给子线程(非主线程)发送消息。 
   
把下面例子中的mHandler声明成类成员,在主线程通过mHandler发送消息即可。        
Android官方文档中Looper的介绍: Class used to run a message loop for a thread.
Threads by default do not have a message loop
associated with them; to create one, call prepare() in the thread that
is to run the loop, and then loop() to have it process messages until
the loop is stopped.
Most interaction with a message loop is through the Handler class.

This
is a typical example of the implementation of a Looper thread, using
the separation of prepare() and loop() to create an initial Handler
to communicate with the Looper.

 1 class LooperThread extends Thread
 2 {
 3 public Handler mHandler;
 4 public void run()
 5 {
 6 Looper.prepare();
 7 mHandler = new Handler()
 8 {
 9 public void handleMessage(Message msg)
10 {
11 // process incoming messages here
12 }
13 };
14 Looper.loop();
15 }

如果线程中使用Looper.prepare()和Looper.loop()创建了消息队列就可以让消息处理在该线程中完成。

android HandlerThread使用小例

之前研究过handler 和 looper 消息队列,不过android里的handler不是另外开启线程来执行的,还是在主UI线程中,如果想另启线程的话需要用到HandlerThread 来实现。在使用HandlerThread的时候需要实现CallBack接口以重写handlerMessage方法,在handlerMessage 方法中来处理自己的逻辑。下来给出一个小例子程序。

layout文件很简单,就一个按钮来启动HanlderTread线程

 1 <?xml version="1.0" encoding="utf-8"?>
 2 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
 3     android:layout_width="fill_parent"
 4     android:layout_height="fill_parent"
 5     android:orientation="vertical" >
 6
 7     <TextView
 8         android:layout_width="fill_parent"
 9         android:layout_height="wrap_content"
10         android:text="@string/hello" />
11
12     <Button
13         android:id="@+id/handlerThreadBtn"
14         android:layout_width="wrap_content"
15         android:layout_height="wrap_content"
16         android:text="startHandlerThread" />
17
18 </LinearLayout>
 1 package com.tayue;
 2
 3 import android.app.Activity;
 4 import android.os.Bundle;
 5 import android.os.Handler;
 6 import android.os.Handler.Callback;
 7 import android.os.HandlerThread;
 8 import android.os.Message;
 9 import android.view.View;
10 import android.view.View.OnClickListener;
11 import android.widget.Button;
12 /**
13  *
14  * @author xionglei
15  *
16  */
17 public class TestHandlerActivity extends Activity implements OnClickListener{
18
19     public Button handlerThreadBTN;
20     MyHandlerThread handlerThread;
21     Handler handler;
22
23     /** Called when the activity is first created. */
24     @Override
25     public void onCreate(Bundle savedInstanceState) {
26         super.onCreate(savedInstanceState);
27         //打印UI线程的名称
28         System.out.println("onCreate  CurrentThread = " + Thread.currentThread().getName());
29
30         setContentView(R.layout.main);
31
32         handlerThreadBTN = (Button) findViewById(R.id.handlerThreadBtn);
33         handlerThreadBTN.setOnClickListener(this);
34
35         handlerThread = new MyHandlerThread("myHanler");
36         handlerThread.start();
37         //注意: 这里必须用到handler的这个构造器,因为需要把callback传进去,从而使自己的HandlerThread的handlerMessage来替换掉Handler原生的handlerThread
38         handler = new Handler(handlerThread.getLooper(), handlerThread);
39     }
40
41     @Override
42     public void onClick(View v) {
43         //点击按钮后来开启线程
44         handler.sendEmptyMessage(1);
45     }
46
47     private class MyHandlerThread extends HandlerThread implements Callback {
48
49         public MyHandlerThread(String name) {
50             super(name);
51         }
52
53         @Override
54         public boolean handleMessage(Message msg) {
55             //打印线程的名称
56             System.out.println(" handleMessage CurrentThread = " + Thread.currentThread().getName());
57             return true;
58         }
59     }
60 }

点击按钮,打印的日志如下(这里点击了3次) 07-06 09:32:48.776: I/System.out(780): onCreate  CurrentThread = main 07-06 09:32:55.076: I/System.out(780):  handleMessage CurrentThread = myHanler 07-06 09:32:58.669: I/System.out(780):  handleMessage CurrentThread = myHanler 07-06 09:33:03.476: I/System.out(780):  handleMessage CurrentThread = myHanler

HandlerThread就这么简单。

当然 android自己也有异步线程的handler,就是AsyncTask,这个类就是封装了HandlerThread 和handler来实现异步多线程的操作的。

同样可以这样使用:

 1 private boolean iscancel = false; //用户手动取消登录的标志位
 2
 3     handlerThread = new HandlerThread("myHandlerThread");
 4                     handlerThread.start();
 5                     handler = new MyHandler(handlerThread.getLooper());
 6                 // 将要执行的线程对象添加到线程队列中
 7                         handler.post(new Runnable() {
 8                             @Override
 9                             public void run() {
10                                 Message message = handler.obtainMessage();
11                                 UserBean user = Bbs.getInstance().Login(username, password);//耗时任务
12                                 Bundle b = new Bundle();
13                                 b.putSerializable("user", user);
14                                 message.setData(b);
15                                 message.sendToTarget(); //或使用 handler.sendMessage(message);
16                             }
17                         });
18     class MyHandler extends Handler {
19
20             public MyHandler(Looper looper) {
21                 super(looper);
22             }
23
24             @Override
25             public void handleMessage(Message msg) {
26                 if(iscancel == false){
27                     // 操作UI线程的代码
28                     Bundle b = msg.getData();
29                     UserBean user = (UserBean)b.get("user");
30                                      ......
31                }
32            }
33     }  
时间: 2024-10-25 18:49:25

【转】Android -- Looper.prepare()和Looper.loop()的相关文章

Android -- Looper.prepare()和Looper.loop() —深入版

Android中的Looper类,是用来封装消息循环和消息队列的一个类,用于在android线程中进行消息处理.handler事实上能够看做是一个工具类,用来向消息队列中插入消息的. (1) Looper类用来为一个线程开启一个消息循环.     默认情况下android中新诞生的线程是没有开启消息循环的.(主线程除外,主线程系统会自己主动为其创建Looper对象,开启消息循环.)     Looper对象通过MessageQueue来存放消息和事件.一个线程仅仅能有一个Looper,相应一个M

Looper.prepare()和Looper.loop()

什么时候需要 Looper Looper用于封装了android线程中的消息循环,默认情况下一个线程是不存在消息循环(message loop)的,需要调用Looper.prepare()来给线程创建一个消息循环,调用Looper.loop()来使消息循环起作用,使用Looper.prepare()和Looper.loop()创建了消息队列就可以让消息处理在该线程中完成. 使用Looper需要注意什么 写在Looper.loop()之后的代码不会被立即执行,当调用后mHandler.getLoo

Android 线程 Looper.prepare()、Looper.loop() 使用

优化项目过程中发现了一个很Low的问题,整理一下,备忘: 说问题之前先看下HandlerThread的定义 一个封装了looper的线程: Looper用于封装了android线程中的消息循环,默认情况下一个线程是不存在消息循环(message loop)的,需要调用Looper.prepare()来给线程创建一个消息循环,调用Looper.loop()来使消息循环起作用,从消息队列里取消息,处理消息. 注:写在Looper.loop()之后的代码不会被立即执行,当调用后mHandler.get

Android java.lang.RuntimeException: Can&#39;t create handler inside thread that has not called Looper.prepare()

E/AndroidRuntime(7200): Uncaught handler: thread Thread-8 exiting due to uncaught exceptionE/AndroidRuntime( 7200): java.lang.RuntimeException: Can't create handler inside thread that has not called Looper.prepare() 原因是非主线程中默认没有创建Looper对象,需要先调用Looper

android Looper.prepare()(转载)

Looper用于封装了android线程中的消息循环,默认情况下一个线程是不存在消息循环(message loop)的,需要调用Looper.prepare()来给线程创建一个消息循环,调用Looper.loop()来使消息循环起作用,从消息队列里取 消息,处理消息. 注:写在Looper.loop()之后的代码不会被立即执行,当调用后 mHandler.getLooper().quit()后,loop才会中止,其后的代码才能得以运行.Looper对象通过MessageQueue 来存放消息和事

关于子线程使用Toast报错Can&#39;t create handler inside thread that has not called Looper.prepare()的解决办法

形同如下代码,在Thread中调用Toast显示错误信息: new Thread(new Runnable(){ @Override public void run() { try{ weatherData = getWeatherData(strUrl); parseJson(weatherData); }catch(Exception e){ Toast.makeText(WindowApplication.getAppContext(), e.toString(), Toast.LENGT

解决使用Handler时Can&#39;t create handler inside thread that has not called Looper.prepare()

在android开发中,主线程不能进行耗时操作,所以我们经常把耗时操作放在子线程中进行,那么就需要子线程与主线程相互交流,就需要使用到handler. 而在使用handler过程中,如果对handler使用不太熟练的话就偶尔会出现Can't create handler inside thread that has not called Looper.prepare()的报错异常.之前在Handler的原理的博文中有讲到,Handler的使用会依靠Looper循环来发送消息,如果不创建Loope

Android : Can&#39;t create handler inside thread that has not called Looper.prepare()

又报错了,不过早也习以为常了. Can't create handler inside thread that has not called Looper.prepare() 我把文档给摘录下来了,大家可以看看. 这个类被用于为线程运行消息循环.默认线程并没有消息循环与之关联,所以你需要创建一个,在线程中调用prepare()以运行这个循环,然后调用loop()在循环结束时获取进程信息. 和消息循环交互最多的就是通过Handler类. 下面是一个实现了Looper线程的典型实例,通过分离的pre

Android Exception 13(Can&#39;t create handler inside thread that has not called Looper.prepare())

10-12 17:02:55.500: E/AndroidRuntime(28343): FATAL EXCEPTION: Timer-2 10-12 17:02:55.500: E/AndroidRuntime(28343): java.lang.RuntimeException: Can't create handler inside thread that has not called Looper.prepare() 10-12 17:02:55.500: E/AndroidRuntim