今天讲一下,在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;public class MainActivity extends ActionBarActivity { private String MyTag = "MyTag"; private int num = 0; private TextView tvObj; private Button btnObj; @Override protected void onCreate(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() { @Override public void onClick(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; @Override public void run() { super.run(); Looper.prepare(); handler = new Handler() { @Override public void handleMessage(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:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" android:paddingBottom="@dimen/activity_vertical_margin" android:gravity="center" android:orientation="vertical" tools:context=".MainActivity"> <TextView android:textSize="30sp" android:layout_width="wrap_content" android:layout_height="wrap_content" android:id="@+id/tvid" /> <Button android:layout_width="match_parent" android:layout_height="wrap_content" android:text="主线程向子线程发送消息" android:textSize="25sp" android:id="@+id/btnid" /></LinearLayout>
最后附上源码的链接(百度云盘,本人使用Android studio编写代码):
http://pan.baidu.com/s/1dDH84r3