Android多线程(二)
在上一篇中,我简单说了用AsyncTask来完成简单异步任务,但AsyncTask是把所有的异步任务放到一个队列中依次在同一个线程中执行。这样就带来一个问题,它无法处理那些耗时长、需要并行的的任务。如何处理这个难题呢?一是自己开启线程然后处理线程通信问题,二是使用HandlerThread这一便捷类来处理。万变不离其宗,先来说明Android线程、及线程通信的原理,然后对于那些便捷的API自然就懂了。
二、Thread 与 Handler
本节涉及的概念较多,有Thread,Handler,Looper,Message,MessageQuene,对于Looper和MessageQueue只是简单的谈及它们在线程通信中的作用,Thread,Handler及Message我会尽力讲清楚些。
1.Thread基础:
1)参考文档:http://developer.android.com/reference/java/lang/Thread.html
我包括我的好些朋友学东西时都会忽略官方文档的重要性,其中很大的原因是因为英文的缘故吧,但其实看懂文档要求英语能力并不高。看别人写的文章、博客、书籍,听别人讲技术,那始终是经过他人过滤后的知识,而这一切未必是你需要的。前车之鉴固然重要,但并不是每一个老人的话都得听。歪果仁写东西有个特点就是他不仅把原理给你讲清楚了,还爱举列子,该怎么做不建议怎么做都会涉及。说了这么多,就是建议自己去看看Android的开发文档,从Guide到Reference内化出自己的知识体系。
2)简单介绍:
看Android的Thread继承树,我们就知道它完全继承了Java的线程体系,在这就不赘言过多的Thread细节了,我就如何开启一个线程讲一讲。开启一个新线程有两种方法,第一种是拓展Thread类,在子类中重写run()方法;第二种是声明Thread对象时传入一个Runnable对象,因为只涉及到一个抽象方法,Runnable对象可以用jdk8的Lambda表达式来代替,这样使得线程的创建变得更加简单。创建好了线程后,如何让线程开始执行呢?调用Thread.run()即可让线程进入待执行队列,当获得CPU时间时它就运行起来了。具体的用法例子会在后面的示例中展现。
2.Handler基础:
1)参考文档:http://developer.android.com/reference/android/os/Handler.html
2)概述:
之前看过一部美剧《天蝎计划》,在介绍团队负责人时,这样说道”He is our goverment handler“。Android中的Handler就是这样一个概念,它是线程通信的发送者和处理者,线程要进行通信时会让handler发出相应的消息,通过Looper传递,Handler发出的消息会在目的线程中得以执行。再举个栗子吧,这是我在《Android编程权威指南》中看到的,它是这么描述线程通信的:两个线程的通信就好像现实中的两个人通过信件来通信,消息队列(MessageQueue)相对于通信时候的信箱;Message(消息)相当于信件;Looper相当于邮递员,它是MessageQueue的操作者;Handler时线程通信的发出者和处理者;每当Thread想要进行通信,它会让Handler投递一个Message给相应的MessageQueue,Looper会一直循环将MessageQueue里的Message发向它的目的地,到达目的地后Looper通知相应的Handler来处理消息。
每一个Handler都是和唯一的Looper对象绑定的,也就是说一个Handler既仅可以个一个Looper绑定,但一个Looper可以有好几个Handler与之关联。Looper操作的MessageQueue是Handler取得消息进行处理和发出消息的地方。
3)使用介绍:
前面说过每一个Handler都是和特别的Looper绑定好的,同时Handler又是处理消息的地方,所以Handler中既要说明和哪个Looper绑定,又要告知怎么处理消息。所以Handler有4个构造方法,下面我来一一介绍:
- Handler()这是无参数构造方法,它默认和当前线程的Looper绑定,未指定消息的处理方式。
- Handler(Looper looper)它需要一个Looper对象作为参数传入,构成出来的Handler对象将和给定的Looper对象绑定。
- Handler(Handler.Callback callback)它需要一个Handler.Callback对象作为参数传入,构造处理的对象和当前线程的Looper绑定并用传入的Handler.Callback对象来处理消息。
- Handler(Looper looper,(Handler.Callback callback)这是第二种和第三种的结合,构成出一个和指定Looper绑定并用指定Callback的回调方法处理消息的Handler对像
- 还有一种使用Handler的方法就是自己拓展Handler类,在子类中实现handlerMessage(Message msg)(这就是接口Callback中的抽象方法),这也是我用的比较多的方式。
Handler的使用就是发送和处理消息,处理消息是在Callback接口中定义好的,当Looper操作的MessageQueue中有消息时,Looper对通知所有与它绑定的Handler调用handlerMessage(Message msg)去处理消息。那怎么发送消息呢?Hander中有一个方法叫sendMessage(Message msg),当调用这个方法后Handler就往Looper操作的MessageQueue中投递一个Message对象,发送消息就算是完成了。简单吧?so easy!
关于Handler的其他方法还请查看文档,将主干的部分弄清楚了,指端末节随着使用慢慢就熟络了。一口气吃不成胖子。对了,下面的Message部分还会提及Handler的一些内容。
3.Message基础:
1)参考文档:http://developer.android.com/reference/android/os/Message.html
2)使用介绍:
线程通信既然叫通信肯定有一个消息的载体,Message就是这个消息的载体,它包含了线程想要通信的全部内容,比如线程运行后得到的结果就和可以包含在Message中。关于Message的构造本来可以按正常的方式构造(就是new一个Message对象,词穷不知道怎么说,就叫它”正常的方式“O(∩_∩)O~),但官方推荐的做法是通过Handler.obtainMessage()获得Message对象,因为这样的Message是从即将被回收的Message中取得的,会避免GC的反复运行和减少运行时的内存消耗。
Message是消息的携带者,它有许多的携带消息的方法,比如setData(Bundle),Message.obj,Message.arg1,Message.arg2等等,需要特别说明的是Message里有一个公开的整形的全局变量what,即Message.what,它一般用来阐述消息的类型,Handler的handlerMessage(Message msg)通常会先检验Message的what变量,然后在决定如何处理。毕竟一个应用中和主线程通信的不可能只用一个线程,一种消息。
4.Looper与MessageQueue简单介绍:
1)参考文档:
Looper:http://developer.android.com/reference/android/os/Looper.html
MessageQueue:http://developer.android.com/reference/android/os/MessageQueue.html
2)简单介绍:
需要注意的地方就是,一个普通Thread创建时是没有Looper对象和它关联的,我们必须在线程的创建中进行关联,具体做法就是在Thread的创建时调用Looper.prepare()进行绑定,调用Looper.loop()使得与线程绑定的Looper对象开始工作。Looper中有一个巨好用的方法,Looper.getMainLooper(),这个方法直接返回当前应用的UI线程的Looper对象,有了Looper对象就可以往主线程发消息了,一会我在示例中会用到这样方法。
关于MessageQueue,其实在线程通信中我们并不直接使用它,只需知道我们通过Handler发送出去的消息都是放在它里的就行了,它是一个”第层次“的对象,我们也不能直接往它里添加消息。但对于理解整个线程通信过程还是很重要的。
5.实践——示例
说了这么多,是时候用实践检验了!先说说我打算怎么做吧!我打算从UI线程向非UI线程(是不是叫主线程和子线程更好?)发一个消息,然后在非UI线程中处理这个消息(这里我只是打印一下日志),然后从非UI线程向主线程发送一个消息,然后在UI线程中处理这个消息(也是简单的打印一下)。好像有些偷懒,但真正使用也大致是这样的,就是把打印换成具体的任务。好了,开始吧!
先给出布局吧!
1 <?xml version="1.0" encoding="utf-8"?> 2 <RelativeLayout 3 xmlns:android="http://schemas.android.com/apk/res/android" 4 xmlns:tools="http://schemas.android.com/tools" 5 android:layout_width="match_parent" 6 android:layout_height="match_parent" 7 android:paddingBottom="@dimen/activity_vertical_margin" 8 android:paddingLeft="@dimen/activity_horizontal_margin" 9 android:paddingRight="@dimen/activity_horizontal_margin" 10 android:paddingTop="@dimen/activity_vertical_margin" 11 tools:context=".MainActivity"> 12 13 <TextView 14 android:layout_width="wrap_content" 15 android:layout_height="wrap_content" 16 android:layout_alignParentTop="true" 17 android:layout_centerHorizontal="true" 18 android:text="Thread和Handler"/> 19 20 <Button 21 android:id="@+id/send_message" 22 android:layout_width="match_parent" 23 android:layout_height="wrap_content" 24 android:layout_alignParentBottom="true" 25 android:text="发送消息"/> 26 27 <Button 28 android:id="@+id/startThread" 29 android:layout_width="match_parent" 30 android:layout_height="wrap_content" 31 android:layout_above="@id/send_message" 32 android:text="开启子线程"/> 33 34 </RelativeLayout>
布局很简单,就是一个textview和两个button,一个button开启线程,并接受来自子线程的信息,另一个从主线程发出消息给子线程。
下面给出代码逻辑:
第一部分是子线程的代码:
1 package comfallblank.github.threadandhandler; 2 3 import android.os.Handler; 4 import android.os.Looper; 5 import android.os.Message; 6 import android.util.Log; 7 8 /** 9 * Created by fallb on 2015/10/7. 10 */ 11 public class MyThread extends Thread { 12 public static final int MSG_WORKER_THREAD = 100; 13 private static final String TAG = "MyThread"; 14 15 private Handler mWorkerHandler; 16 private Handler mMainHandler; 17 18 public MyThread(Handler handler) { 19 mMainHandler = handler; 20 mWorkerHandler = new Handler(){ 21 @Override 22 public void handleMessage(Message msg) { 23 if(msg.what == MainActivity.MSG_MAIN){ 24 Log.d(TAG,"Message:"+msg.obj); 25 } 26 } 27 }; 28 } 29 30 @Override 31 public void run() { 32 Looper.prepare(); 33 Message msg = mMainHandler.obtainMessage(); 34 msg.what = MyThread.MSG_WORKER_THREAD; 35 msg.obj="子线程发出的消息"; 36 mMainHandler.sendMessage(msg); 37 38 Looper.loop(); 39 } 40 41 public Handler getWorkerHandler() { 42 return mWorkerHandler; 43 } 44 }
第二份是主线程的:
1 package comfallblank.github.threadandhandler; 2 3 import android.os.Bundle; 4 import android.os.Handler; 5 import android.os.Message; 6 import android.support.v7.app.AppCompatActivity; 7 import android.util.Log; 8 import android.view.View; 9 import android.widget.Button; 10 11 public class MainActivity extends AppCompatActivity { 12 public static final int MSG_MAIN = 100; 13 private static final String TAG = "MainActivity"; 14 15 private Button mStartThread; 16 private Button mSendMessage; 17 private Handler mHandler = new MyHandler(); 18 19 @Override 20 protected void onCreate(Bundle savedInstanceState) { 21 super.onCreate(savedInstanceState); 22 setContentView(R.layout.activity_main); 23 24 final MyThread thread = new MyThread(mHandler); 25 mStartThread = (Button) findViewById(R.id.startThread); 26 mStartThread.setOnClickListener(new View.OnClickListener() { 27 @Override 28 public void onClick(View view) { 29 Log.d(TAG, "开启线程"); 30 thread.start(); 31 } 32 }); 33 mSendMessage = (Button) findViewById(R.id.send_message); 34 mSendMessage.setOnClickListener(new View.OnClickListener() { 35 @Override 36 public void onClick(View view) { 37 Handler workerHandler = thread.getWorkerHandler(); 38 Message msg = workerHandler.obtainMessage(); 39 msg.what = MSG_MAIN; 40 msg.obj = "来自主线程的消息"; 41 workerHandler.sendMessage(msg); 42 } 43 }); 44 45 } 46 47 class MyHandler extends Handler { 48 @Override 49 public void handleMessage(Message msg) { 50 if (msg.what == MyThread.MSG_WORKER_THREAD) { 51 Log.d(TAG,"Message:"+msg.obj); 52 } 53 } 54 } 55 }
Logcat打印日志如下:
10-07 19:34:34.424 19671-19671/comfallblank.github.threadandhandler D/MainActivity: 开启线程
10-07 19:34:34.454 19671-19671/comfallblank.github.threadandhandler D/MainActivity: Message:子线程发出的消息
10-07 19:34:37.267 19671-19671/comfallblank.github.threadandhandler D/MyThread: Message:来自主线程的消息
6.关于Handler再说两句:
Handler对象除了发出消息外,还有一族方法,来发出一个Runnable对象,Handler.post(Runnable runnable)及时其中的一个,它向MessageQueue中添加这个Runnable对象,然后目的线程的MessageQueue会执行该方法。
关于Thread、Handler的内容就说到这里了,还是如前上一篇所说,本文只起到抛砖引玉的作用,多有偏颇,往交流指正。