今天讲一下,在Android中主线程如何向子线程中发送消息的问题。 或许回想无非就是创建一个Handler对象,然后一个线程发消息,另一个接收消息嘛…… 原理确实是这样,但是我们平时,是从子线程向主线程发消息,而主线程默认已经帮我们完成了Looper的操作,所以我们只需要简单的“创建一个Handler对象,然后一个线程发消息,另一个接收消息”…… 我们先说一下这个Looper是神马吧。 它就像一个消息队列(MessageQueue)的管家(Looper),一个消息队列只有一个管家,并且管理者整个消息队列,,而一个消息队列内可以容纳多个消息(Message),而工人(Handler)也可以有多个,管家派遣他们向消息队列中存储或取出消息后执行任务;
所以它们的个数如下: 管家(Looper): 1 个; 消息队列(MessageQueue):1 个; 消息(Message):可以多个; 工人(Handler):可以多个;
对Looper的Android源码整理了一下,可以得到如下的内容:
//ThreadLocal可以理解为一个存储Looper的列表;ThreadLocal<Looper> sThreadLocal = new ThreadLocal<Looper>();//Looper的构造器private Looper(boolean quitAllowed) {mQueue = new MessageQueue(quitAllowed);mThread = Thread.currentThread();}1.Looper.prepare()完成了如下操作:sThreadLocal.set(new Looper(quitAllowed));2.Looper.loop()完成了如下操作:Looper me = sThreadLocal.get();MessageQueue queue = me.mQueue;……//不断循环分发消息for (;;) {Message msg = queue.next();……msg.target.dispatchMessage(msg);//dispatchMessage(msg)方法执行如下内容:handler.handleMessage(msg);//handler执行handleMessage(msg)中的代码;……};
下面写一个小例子: 一个Button被按下时,从主线程向子线程发送一个数字,然后子线程将数字用Toast显示,而主线程将TextView也被设置成该数字;
运行截图如下:
接下来附上代码:
MainActivity.java:
package activity.wyc.com.looperthreaddemo;import android.os.Handler;import android.os.Looper;import android.os.Message;import android.support.v7.app.ActionBarActivity;import android.os.Bundle;import android.util.Log;import android.view.Menu;import android.view.MenuItem;import android.view.View;import android.widget.Button;import android.widget.TextView;import android.widget.Toast;{private String MyTag = “MyTag”;private int num = 0;private TextView tvObj;private Button btnObj;(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);tvObj = (TextView) findViewById(R.id.tvid);btnObj = (Button) findViewById(R.id.btnid);final LooperThread looperThread = new LooperThread();looperThread.start();btnObj.setOnClickListener(new View.OnClickListener() {(View v) {Message message = Message.obtain();message.arg1 = num;tvObj.setText(“主线程发送了 :”+String.valueOf(message.arg1));looperThread.handler.sendMessage(message);num++;}});}class LooperThread extends Thread {public Handler handler;() {super.run();Looper.prepare();handler = new Handler() {(Message msg) {super.handleMessage(msg);Toast.makeText(MainActivity.this,”LooperThread handler 收到消息 :”+msg.arg1,Toast.LENGTH_LONG).show();Log.i(MyTag, “LooperThread handler 收到消息 :” + msg.arg1);}};Looper.loop();//loop()会调用到handler的handleMessage(Message msg)方法,所以,写在下面;}}}
activity_main.xml:
=====”vertical”tools:context=”.MainActivity”><TextView=”wrap_content”android:id=”@+id/tvid” /><Button==”@+id/btnid” /></LinearLayout>
最后附上源码的链接(百度云盘,本人使用Android studio编写代码):
生命中,每一种苦难的背后都有一片晴朗的天空