91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

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

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

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

修水县| 鹤壁市| 灵武市| 漳平市| 黄浦区| 康保县| 镇巴县| 潜山县| 沅江市| 肃北| 长海县| 渝北区| 丰台区| 大安市| 锦州市| 洞口县| 甘洛县| 聂拉木县| 高清| 庄浪县| 收藏| 木里| 新源县| 威信县| 常德市| 禹城市| 友谊县| 积石山| 慈利县| 青田县| 乌什县| 凤山市| 九江县| 太保市| 仪征市| 弥渡县| 湖南省| 会宁县| 青州市| 昌宁县| 夏河县|