Android:主线程如何向子线程发送消息

今天讲一下,在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

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。