问题描述
android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
at android.view.ViewRootImpl.checkThread(ViewRootImpl.java:7820)
at android.view.ViewRootImpl.invalidateChildInParent(ViewRootImpl.java:1349)
at android.view.ViewGroup.invalidateChild(ViewGroup.java:5446)
at android.view.View.invalidateInternal(View.java:14755)
at android.view.View.invalidate(View.java:14719)
at android.view.View.invalidate(View.java:14703)
发生问题时的截图
问题分析
android项目开发中有时候会发生类似的问题,这是一个典型的线程安全问题。Only the original thread that created a view hierarchy can touch its views解释为只有创建视图层次结构的原始线程才能触及它的视图,实际上是说只有主线程才可以操作UI状态的变化。
解决方法
方法(一)
runOnUiThread(new Runnable() {
@Override
public void run() {
//TODO 需要在UI线程上操作的方法
}
});
方法(二)
new Handler().post(new Runnable() {
@Override
public void run() {
//TODO 需要在UI线程上操作的方法
}
});
方法(三)
Handler handler = new Handler();
Runnable runnable = new Runnable() {
@Override
public void run() {
//TODO 需要在UI线程上操作的方法
}
};
改方法跟方法二是一样的,只是写法不同。只需要在调用的地方:
handler.post(runnable);
方法(四)
Handler handler = new Handler(){
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
if(msg.what==0x10){
//TODO 需要在UI线程上操作的方法
}
}
};
然后,在需要的地方调用如下方法就行
Message message = handler.obtainMessage();
message.what=0x10;
handler.sendMessage(message);
总结:上述四种方法都可以将UI操作由子线程转到主线程去处理。其中方法(一)和方法(二)比较简单方便,推荐使用。