今天讲一下,在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编写代码):