Android《多线程-中》

上篇中我们讲解了线程,进程以及Thread和Runnable之间的区别,那么这一篇我们来讲解下Android应用的消息处理机制,之后才能够更深刻的了解为什么多线程能够解决UI县城阻塞的问题。

Android 的消息处理机制

说到Android消息处理机制有的人或许有些概念模糊,那么Handler、Looper、MessageQueue,大家应该比较面熟吧。

  • UI线程
    我们知道在Android应用启动时,会默认启动一个(主)UI线程,这个线程会关联一个消息队列所有的操作都会被封装成消息交给主线程来处理。为了保证主线程不会主动退出,就要将抓取消息的操作放在一个死循环中,这样我们的程序就不会主动退出并保持运行状态。
    那么和Handler 、 Looper 、Message有啥关系?其实Looper负责的就是创建一个MessageQueue,然后进入一个无限循环体不断从该MessageQueue中读取消息,而消息的创建者就是一个或多个Handler 。

上面虽然是原理但是并不是多好理解,我们呢打个比方:
男主:BOY
女主:Girl
当Boy和Girl 结婚后(APP启动了),那么Boy就要开始干活了(APP开启UI线程),这时候Boy就要进入疯狂工作模式了 (无限循环-Looper),而我们的Girl(Handler的一个实例)会把各种类型的账单记下来(将Message写入MessageQueue),但凡Boy接收到(Looper获取)账单(Message)就进行处理,如果是月初Girl没有给你霍霍,那么Boy就会自己攒钱了(阻塞Looper)等到月底账单来临再去处理。

举例讲完后我们来看看Handler、Looper、MessageQueue的概念。

  • Handler:
    简单说Handler用于同一个进程的线程间通信,另外一个作用,就是能统一处理消息的回调。这样一个Handler发出消息又确保消息处理也是自己来做,这样的设计非常的赞。
  • Looper:
    无限循环不退出的线程,Looper的另外一部分工作就是在循环代码中会不断从消息队列挨个拿出消息给主线程处理。
  • MessageQueue
    MessageQueue 存在的原因很简单,就是同一线程在同一时间只能处理一个消息,同一线程代码执行是不具有并发性,所以需要队列来保存消息和安排每个消息的处理顺序。

这里就不带大家去看源代码了。
下面我峨嵋你通过一个例子来看下消息处理机制是怎么解决网络加载时阻塞UI线程问题的。

案例中我们让下载类 sleep 7秒,并且下载前和下载后都要对UI中的Text进行修改。

MainActivity.class

/**
 * Created by 泅渡者
 * Created on 2017/10/27.
 */

public class MainActivity extends AppCompatActivity {
    private static final String TAG = "MainActivity";

    private final Handler mHandler = new MyHandler(this);

    public static TextView tv_download;
    private Message message;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        tv_download = findViewById(R.id.tv_download);

        tv_download.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Download download = new Download();
                Thread thread = new Thread(download,"下载");
                thread.start();
            }
        });
    }


    private static class MyHandler extends Handler {

        private final WeakReference<Activity> mActivity;

        public MyHandler(Activity activity) {
            mActivity = new WeakReference<Activity>(activity);


        }

        @Override
        public void handleMessage(Message msg) {
            switch (msg.what){
                case 1:
                    if (mActivity.get() == null) {
                        return;
                    }
                    tv_download.setText("下载中。。。");
                    break;
                case 2:
                    if (mActivity.get() == null) {
                        return;
                    }
                    tv_download.setText("下载完成");
                    break;
                    default:
                        return;
            }

        }
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();

        mHandler.removeCallbacksAndMessages(null);

    }

    class Download implements Runnable{

        @Override
        public void run() {

            try {
                message= mHandler.obtainMessage();
                message.what = 1;
                mHandler.sendMessage(message);

                Thread.sleep(7000);

                message= mHandler.obtainMessage();
                message.what = 2;
                mHandler.sendMessage(message);

            } catch (InterruptedException e) {

                Log.e(TAG,e.toString());

            }
        }
    }
}

布局文件

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
    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"
    tools:context="com.bsoft.multithread.MainActivity">

    <TextView
        android:id="@+id/tv_download"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerInParent="true"
        android:text="下载"
       />

</RelativeLayout>

运行效果如下:

2.gif

我们看下面代码:

                message= mHandler.obtainMessage();
                message.what = 1;
                mHandler.sendMessage(message);

                Thread.sleep(7000);

                message= mHandler.obtainMessage();
                message.what = 2;
                mHandler.sendMessage(message);

这里我们在子线程不能操作UI线程,这个大家都知道,我们说一下obtainMessage()。

  1. obtainmessage()是从消息池中拿来一个msg 不需要另开辟空间。
  2. new Message()需要重新申请,效率低。
  3. obtianmessage可以循环利用。

这里还有一个比较重要的话题,就是由Handler导致Activity 的内存泄露。

Handler内存泄漏解决办法

Handler也是造成内存泄露的一个重要的源头,主要Handler属于TLS(Thread Local Storage)变量,生命周期和Activity是不一致的,Handler引用Activity会存在内存泄露。
我们一般用法是否是这样的呢?

  private final Handler mHandler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            switch (msg.what){
                case 1:
                    //TODO
                    break;
                default:
                    return;
            }

        }
    };
------------------------------------------------------------------------------
        Message message = mHandler.obtainMessage();
        message.what = 1;
        mHandler.sendMessageDelayed(message,6000);

但是程序会提示

This Handler class should be static or leaks might occur (anonymous android.os.Handler)
意思:此处理程序类应该是静态的或可能发生泄漏 (匿名 android.os.Handler)

是什么导致的呢 ?

  • 生命周期
    Handler 的生命周期与Activity 不一致,当Android应用启动的时候,会先创建一个UI主线程的Looper对象,Looper实现了一个简单的消息队列,一个一个的处理里面的Message对象。主线程Looper对象在整个应用生命周期中存在,当在主线程中初始化Handler时,该Handler和Looper的消息队列关联(没有关联会报错的)。发送到消息队列的Message会引用发送该消息的Handler对象,这样系统可以调用 Handler#handleMessage(Message) 来分发处理该消息。

  • handler 引用 Activity 阻止了GC对Acivity的回收
    在Java中,非静态(匿名)内部类会默认隐性引用外部类对象。而静态内部类不会引用外部类对象。如果外部类是Activity,则会引起Activity泄露 ,当Activity finish后,延时消息会继续存在主线程消息队列中1分钟,然后处理消息。而该消息引用了Activity的Handler对象,然后这个Handler又引用了这个Activity。这些引用对象会保持到该消息被处理完,这样就导致该Activity对象无法被回收,从而导致了上面说的 Activity泄露。

  • 如何避免?
    使用显形的引用:1.静态内部类 2. 外部类
    使用弱引用 : WeakReference
    还要在程序销毁时进行remove();

@Override  
public void onDestroy() {  
    mHandler.removeCallbacksAndMessages(null);  
}  

上述案例就是应用弱引用,可能大家觉得写的代码比较多,不怕我们有办法。我们按照图的指示来创建自己的Live Templates:

2.gif

OK 今天的就到这里。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,717评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,501评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,311评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,417评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,500评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,538评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,557评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,310评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,759评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,065评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,233评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,909评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,548评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,172评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,420评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,103评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,098评论 2 352

推荐阅读更多精彩内容