HandlerThread 使用

在Android中,主線程和子線程通信,最簡單最直接的方式就是使用Handler和Message,在子線程中使用 在主線程中創建的 Handler對象發送一個msg,就可以把子線程的消息發送到主線程中進行操作,這個在平常開發中很常見。
現在來想另外一種使用場景,主線程往子線程里,發送一個消息,讓子線程來做不同的操作。
其實也可以通過Handler 和Msg 來操作。
<b>來模擬一個例子 經理打電話讓司機接人的例子</b>。
經理打電話相當于在主線程發消息,司機在子線程完成接人的功能,然后反饋給經理(主線程)
來看代碼

public class ThreadHandlerAc extends BaseActivity {

    private Button mButton;
    //經理的handler
    private Handler mMangerHandler;
    //司機的handler
    private Handler mDriverHandler;

    private EditText mLog_tv;

    @Override
    protected void initView() {
        setContentView(R.layout.ac_test);
        mButton = (Button) findViewById(R.id.ac_test_bt);
        mLog_tv=(EditText)findViewById(R.id.ac_test_log);

        driver.start();//司機開始等待
        mButton.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //打電話給司機
                managerCallDriver("老總");

            }
        });

            mMangerHandler = new Handler() {
                @Override
                public void handleMessage(Message msg) {
                    super.handleMessage(msg);
                    log("經理收到接到人消息了 , 發生在 " + Thread.currentThread().getName());
                }
            };


    }

    private void log(final String log){
       runOnUiThread(new Runnable() {
           @Override
           public void run() {
               mLog_tv.setText(mLog_tv.getText() + "\n" + log) ;
           };
       });

    }

    //通知司機去接人
    private void managerCallDriver(String name) {
        log("經理打電話給司機去接"+name  + "發生在 " + Thread.currentThread().getName()+"線程");
        Message managerMsg = new Message();
        managerMsg.obj = name;
        mDriverHandler.sendMessage(managerMsg);
    }


    Thread driver = new Thread("driver") {
        @Override
        public void run() {
            super.run();
            //創建looper
            Looper.prepare();
            //從ThreadLoacl中回去looper
            mDriverHandler = new Handler(Looper.myLooper()) {
                @Override
                public void handleMessage(Message msg) {
                    super.handleMessage(msg);
                    try {
                        Thread.sleep(200);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    log("司機收到消息去接" + msg.obj + "發生在 " + Thread.currentThread().getName()+"線程");
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    log("司機接到人了,通知經理,發生在" + Thread.currentThread().getName()+"線程");

                    Message dreiverMsg = new Message();
                    dreiverMsg.obj = "經理,我接到人了";
                    dreiverMsg.what = 1;
                    mMangerHandler.sendMessage(dreiverMsg);
                }
            };
            //loop循環會讓這個線程一直不會 停
            Looper.loop();

        }
    };


}


 

執行的結果

執行結果

注意代碼Thread里run方法,需要自己去得到一個looper對象,需要自己去開啟loop循環,而且,loop會一直循環,也就說這個線程一直開啟,這樣會導致Activity 無法回收,還得自己去結束線程,這樣太麻煩了,其實可以利用Android 提供的ThreadHanlder類,同樣可以完成,而且還不需要管理里looper,并且提供了結束線程的方法。
代碼

public class ThreadHandlerAc extends BaseActivity {

    private Button mButton;
    //經理的handler
    private Handler mMangerHandler;
    //司機的handler
    private Handler mDriverHandler;

    private EditText mLog_tv;

    private HandlerThread driver;

    @Override
    protected void initView() {
        setContentView(R.layout.ac_test);
        mButton = (Button) findViewById(R.id.ac_test_bt);
        mLog_tv = (EditText) findViewById(R.id.ac_test_log);
        driver = new HandlerThread("driver");
        driver.start();//司機開始等待
        mDriverHandler = new Handler(driver.getLooper() ) {
            @Override
            public void handleMessage(Message msg) {
                super.handleMessage(msg);
                try {
                    Thread.sleep(200);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                log("司機收到消息去接" + msg.obj + "發生在 " + Thread.currentThread().getName() + "線程");
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                log("司機接到人了,通知經理,發生在" + Thread.currentThread().getName() + "線程");
                Message dreiverMsg = new Message();
                dreiverMsg.obj = "經理,我接到人了";
                dreiverMsg.what = 1;
                mMangerHandler.sendMessage(dreiverMsg);
            }
        };

        mButton.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //打電話給司機
                managerCallDriver("老總");

            }
        });

        mMangerHandler = new Handler() {
            @Override
            public void handleMessage(Message msg) {
                super.handleMessage(msg);
                log("經理收到接到人消息了 , 發生在 " + Thread.currentThread().getName());
            }
        };
    }

    private void log(final String log) {
        runOnUiThread(new Runnable() {
            @Override
            public void run() {
                mLog_tv.setText(mLog_tv.getText() + "\n" + log);
            }

            ;
        });

    }

    //通知司機去接人
    private void managerCallDriver(String name) {
        log("經理打電話給司機去接" + name + "發生在 " + Thread.currentThread().getName() + "線程");
        Message managerMsg = new Message();
        managerMsg.obj = name;
        mDriverHandler.sendMessage(managerMsg);
    }

    @Override
    protected void onDestroy() {
        driver.quit();
        super.onDestroy();
    }
}

同樣是完成了功能,還不需要提供looper的管理,并且有直接退出線程方法,是不是方便很多。
(如果博客有錯誤的地方,歡迎留言指正)。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容