中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Android HandlerThread的使用及原理詳解

發布時間:2020-09-02 13:18:16 來源:腳本之家 閱讀:235 作者:Will 欄目:移動開發

一、HandlerThread的含義

HandlerThread能夠新建擁有Looper的線程。這個Looper能夠用來新建其他的Handler。(線程中的Looper)需要注意的是,新建的時候需要被回調。

二、HandlerThread的用法

一般情況下,我們會經常用Handler在子線程中更新UI線程,那是因為在主線程中有Looper循環,而HandlerThread新建擁有Looper的子線程又有什么用呢?

必然是執行耗時操作。舉個例子,數據實時更新,我們每10秒需要切換一下顯示的數據,如果我們將這種長時間的反復調用操作放到UI線程中,雖說可以執行,但是這樣的操作多了之后,很容易會讓UI線程卡頓甚至崩潰。

于是,就必須在子線程中調用這些了。
HandlerThread繼承自Thread,一般適應的場景,便是集Thread和Handler之所長,適用于會長時間在后臺運行,并且間隔時間內(或適當情況下)會調用的情況,比如上面所說的實時更新。

三、實現每2秒更新一下UI

public class MainActivity extends AppCompatActivity {
 
  private TextView tvMain;
 
  private HandlerThread mHandlerThread;
  //子線程中的handler
  private Handler mThreadHandler;
  //UI線程中的handler
  private Handler mMainHandler = new Handler();
 
  //以防退出界面后Handler還在執行
  private boolean isUpdateInfo;
  //用以表示該handler的常熟
  private static final int MSG_UPDATE_INFO = 0x110;
 
  @Override
  protected void onCreate(Bundle savedInstanceState)
  {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
 
    tvMain = (TextView) findViewById(R.id.tv_main);
 
    initThread();
  }
 
 
  private void initThread()
  {
    mHandlerThread = new HandlerThread("check-message-coming");
    mHandlerThread.start();
 
    mThreadHandler = new Handler(mHandlerThread.getLooper())
    {
      @Override
      public void handleMessage(Message msg)
      {
        update();//模擬數據更新
 
        if (isUpdateInfo)
          mThreadHandler.sendEmptyMessage(MSG_UPDATE_INFO);
      }
    };
 
  }
 
  private void update()
  {
    try
    {
      //模擬耗時
      Thread.sleep(2000);
      mMainHandler.post(new Runnable()
      {
        @Override
        public void run()
        {
          String result = "每隔2秒更新一下數據:";
          result += Math.random();
          tvMain.setText(result);
        }
      });
 
    } catch (InterruptedException e)
    {
      e.printStackTrace();
    }
 
  }
 
  @Override
  protected void onResume()
  {
    super.onResume();
    //開始查詢
    isUpdateInfo = true;
    mThreadHandler.sendEmptyMessage(MSG_UPDATE_INFO);
  }
 
  @Override
  protected void onPause()
  {
    super.onPause();
    //停止查詢
    //以防退出界面后Handler還在執行
    isUpdateInfo = false;
    mThreadHandler.removeMessages(MSG_UPDATE_INFO);
  }
 
  @Override
  protected void onDestroy()
  {
    super.onDestroy();
    //釋放資源
    mHandlerThread.quit();
  }
}

四、HandlerThread 原理

public class HandlerThread extends Thread {
  int mPriority;
  int mTid = -1;
  Looper mLooper;
 
  public HandlerThread(String name) {
    super(name);
    mPriority = Process.THREAD_PRIORITY_DEFAULT;
  }
 
 
  public HandlerThread(String name, int priority) {
    super(name);
    mPriority = priority;
  }
 
 
  protected void onLooperPrepared() {
  }
 
  @Override
  public void run() {
    mTid = Process.myTid();
    Looper.prepare();
    synchronized (this) {
      mLooper = Looper.myLooper();
      notifyAll();
    }
    Process.setThreadPriority(mPriority);
    onLooperPrepared();
    Looper.loop();
    mTid = -1;
  }
 
 
  public Looper getLooper() {
    if (!isAlive()) {
      return null;
    }
 
    // If the thread has been started, wait until the looper has been created.
    synchronized (this) {
      while (isAlive() && mLooper == null) {
        try {
          wait();
        } catch (InterruptedException e) {
        }
      }
    }
    return mLooper;
  }
 
 
  public boolean quit() {
    Looper looper = getLooper();
    if (looper != null) {
      looper.quit();
      return true;
    }
    return false;
  }
 
 
  public boolean quitSafely() {
    Looper looper = getLooper();
    if (looper != null) {
      looper.quitSafely();
      return true;
    }
    return false;
  }
 
 
  public int getThreadId() {
    return mTid;
  }
}

首先我們可以看到HandlerThread繼承自Thread,因此在run()中的邏輯都是在子線程中運行的。

接下來就是兩個關鍵的方法,run()和getLooper():
run()中可以看到是很簡單的創建Looper以及讓Looper工作的邏輯。
run()里面當mLooper創建完成后有個notifyAll(),getLooper()中有個wait(),這有什么用呢?因為的mLooper在一個線程中執行創建,而我們的handler是在UI線程中調用getLooper()初始化的。
也就是說,我們必須等到mLooper創建完成,才能正確的返回。getLooper();wait(),notify()就是為了解決這兩個線程的同步問題。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

方城县| 洪湖市| 广灵县| 清新县| 渭南市| 石屏县| 遂昌县| 都兰县| 儋州市| 曲周县| 泰州市| 乐业县| 彰化市| 海伦市| 成武县| 桂东县| 墨脱县| 凤庆县| 和静县| 铁岭市| 高邮市| 襄汾县| 扎兰屯市| 松江区| 凤城市| 龙游县| 喀喇沁旗| 津南区| 小金县| 周宁县| 闻喜县| 金沙县| 麦盖提县| 长顺县| 青田县| 竹北市| 武宁县| 扎囊县| 肥东县| 嘉黎县| 肇东市|