问题:“Android只能在UI线程更新UI 么?”
答:“对!......,嗯?不对?”
我脑子里的的回答是“对”,但是辩证思维又在提醒我可能有陷阱,于是我就说“大部分情况是的”。那么小部分情况呢?具体说不上来了!于是才发现这个问题一直被忽略了。
于是试验检验真理,撸代码验证了一遍。
new Thread(){
@Override
public void run() {
super.run();
btn_demo1.setText("Demo1--"+Thread.currentThread().getName());
}
}.start();
奔溃信息:
CalledFromWrongThreadException
问题出现在ViewRootImpl.checkThread()的时候出错
查看ViewRootImpl的源码,导致问题的原因:
public ViewRootImpl(Context context, Display display) {
mContext = context;
mWindowSession = WindowManagerGlobal.getWindowSession();
mDisplay = display;
mBasePackageName = context.getBasePackageName();
mThread = Thread.currentThread();
.......
}
@Override
public void requestLayout() {
if (!mHandlingLayoutInLayoutRequest) {
checkThread();
mLayoutRequested = true;
scheduleTraversals();
}
}
void checkThread() {
if (mThread != Thread.currentThread()) {
throw new CalledFromWrongThreadException(
"Only the original thread that created a view hierarchy can touch its views.");
}
}
ViewRootImpl$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
CalledFromWrongThreadException提示:只能在创建View的线程里操作view.
那么意思是我在非主线程创建View,就可以在非主线程操作该view了咯!
于是:
private void addWindView(){
TextView tx = new TextView(MainActivity.this);
tx.setText("今天天气很好哦!");
tx.setTextColor(getResources().getColor(R.color.white));
tx.setBackgroundColor(getResources().getColor(R.color.colorPrimary));
tx.setGravity(Gravity.CENTER);
WindowManager wm = MainActivity.this.getWindowManager();
WindowManager.LayoutParams params = new WindowManager.LayoutParams(
250, 150, 200, 200, WindowManager.LayoutParams.FIRST_SUB_WINDOW,
WindowManager.LayoutParams.TYPE_TOAST, PixelFormat.OPAQUE);
wm.addView(tx, params);
}
new Thread(){
@Override
public void run() {
super.run();
addWindView();
}
}.start();
不幸的是,还是崩了。
崩溃信息:
not call Looper.perpare()
源码中:
void checkThread() 通过了,可是在scheduleTraversals()刷新UI的时候:
final ViewRootHandler mHandler = new ViewRootHandler();
.......
void scheduleTraversals() {
if (!mTraversalScheduled) {
mTraversalScheduled = true;
mTraversalBarrier = mHandler.getLooper().getQueue().postSyncBarrier();
mChoreographer.postCallback(
Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);
if (!mUnbufferedInputDispatch) {
scheduleConsumeBatchedInput();
}
notifyRendererOfFramePending();
pokeDrawLockIfNeeded();
}
}
所以没有Looper实例化的异常。
于是加上Looper.perpare(),和Looper.loop().
new Thread(){
@Override
public void run() {
super.run();
Looper.perpare()
addWindView();
Looper.loop()
}
}.start();
这下没有报错并且成功加载显示UI.
所以,Android中非UI主线程能不能操作UI?答案是可以的。只不过只能在创建View的线程里操作view.
总结:
1.由于Android是通过Handler消息机制的方式刷新UI的。所以Android 的UI控件是线程安全的,不会导致多线程访问使得UI处于不可预期的状态。
2.Android每次刷新UI的时候,最终根布局ViewRootImpl.checkThread()来检验线程是否是View的创建线程。
网友评论