Android handler使用方法

android中系統不允許在非主線程更新UI。當我們在非主線程做了耗時操作后,需要去更新UI的時候,我們就需要使用handler來執行更新操作。

首先在Activity中新建一個handler對象,Handler handler = new Handler();這樣我們的handler會自動綁定到ActivityThread線程中。當我們要更新UI的時候可以使用handler.post(runnable)方法,將要執行的操作交給handler執行。
如果我們在一個非UI線程中更新UI將會拋出運行時異常:
android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.

下面是一個使用Handler更新UI的demo:


public class MainActivity extends ActionBarActivity {
private Handler handler = new Handler();
private TextView textView;

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

    textView = (TextView) findViewById(R.id.hello);

    new Thread() {
        @Override
        public void run() {
            super.run();
            try {
                Thread.sleep(3000);
                handler.post(new Runnable() {
                    @Override
                    public void run() {
                        textView.setText("update by handler");
                    }
                });
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }.start();
}
} 

常見的Handler使用方法:

  1. 在主線程中new 一個對象Handler handler = new Handler()(會自動綁定到主線程)。在需要更新的地方使用handler.post(runnable);發送一個runnable對象在runnable的run方法中執行我們想要的操作。
  2. 在主線程new 一個handler 在需要更新的地方使用 handler.sendMessage()方法傳遞一個Message對象。可以使用message對象的arg1參數或者obj參數或者what參數攜帶參數。在handler中的回調handler.handleMessage方法中執行想做的方法。
  3. 在主線程新建一個Runnable對象和Handler對象。在需要子線程中使用handler.post()來更新操作。
    下面是一個demo每隔1一秒更新textView;
public class MainActivity extends ActionBarActivity {
private Handler handler = new Handler();
private TextView textView;
private Runnable runnable = new Runnable() {
    @Override
    public void run() {
        textView.setText(System.currentTimeMillis() + "");
        try {
            Thread.sleep(1000);
            handler.post(runnable);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
};

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

    textView = (TextView) findViewById(R.id.hello);

    new Thread() {
        @Override
        public void run() {
            super.run();
            handler.post(runnable);
        }
    }.start();
}
}

當這個runnable被post出去后會進入無線循環,一直發送runnable。當我們要取消這個不讓其更新怎么辦呢,需要remove這個runnable。現在新建一個Button讓button點擊后停止更新TextView().

在Button的OnClick時間中加入:


@Override

public void onClick(View v) {
    handler.removeCallbacks(runnable);
}

攔截Handler的message;當我們構造一個handler的時候使用Handler handler = new Handler(new Callback(){});的時候來看看效果。

現在我們讓textView自動改變從1開始遞增加,但是增加10的時候就自動停止不再增加了。

public class MainActivity extends ActionBarActivity { 
private int count = 1;
private TextView textView;
private Handler handler = new Handler(new Handler.Callback() {
    @Override
    public boolean handleMessage(Message msg) {
        if (count == 10) {
            return true;  //當我們的Callback對象返回true后就不會再執行Handler的handleMessage了,相當于攔截了handler的處理
        } else {
            return false; //當返回false后不影響
        }
    }
}) {
    @Override
    public void handleMessage(Message msg) {
        super.handleMessage(msg);
        textView.setText(count + "");
        count++;
    }
};

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

    textView = (TextView) findViewById(R.id.hello);

    new Thread() {
        @Override
        public void run() {
            super.run();
            for (int i = 0; i < 50; i++) {
                try {
                    Thread.sleep(1000);
                    handler.sendEmptyMessage(1);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    }.start();
}
}

小結:

Androi更新UI要設計成單線程模型,主要是為了防止多個線程同時修改界面,導致的界面錯亂,雖然也可以通過對界面資源加鎖來控制只讓某個線程在某一時刻只修改一個資源,但是明顯增加了復雜性。而使用單線程模型,加上我們的handler就可用很好的解決這個問題了。新建handler對象的時候就會自動綁定到我們的主線程,然后當handler收到消息后就是在主線程中執行的了。

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

推薦閱讀更多精彩內容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 173,558評論 25 708
  • Android Handler機制系列文章整體內容如下: Android Handler機制1之ThreadAnd...
    隔壁老李頭閱讀 8,267評論 8 57
  • 異步消息處理線程啟動后會進入一個無限的循環體之中,每循環一次,從其內部的消息隊列中取出一個消息,然后回調相應的消息...
    cxm11閱讀 6,455評論 2 39
  • Spring Cloud為開發人員提供了快速構建分布式系統中一些常見模式的工具(例如配置管理,服務發現,斷路器,智...
    卡卡羅2017閱讀 134,992評論 19 139
  • 在這個天下文章一大抄大抄特抄抄了也白抄的網絡,談這個有什么意義?主要是有梗在喉,不吐不快。 在微信平臺上,同追一個...
    思遙閱讀 271評論 0 1