Android中非UI主线程能不能操作UI()

问题:“Android只能在UI线程更新UI 么?”
答:“对!......,嗯?不对?”
我脑子里的的回答是“对”,但是辩证思维又在提醒我可能有陷阱,于是我就说“大部分情况是的”。那么小部分情况呢?具体说不上来了!于是才发现这个问题一直被忽略了。
于是试验检验真理,撸代码验证了一遍。

new Thread(){ @Override public void run() { super.run(); btn_demo1.setText("Demo1--"+Thread.currentThread().getName()); } }.start();

奔溃信息:

Android中非UI主线程能不能操作UI()
文章图片
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();

不幸的是,还是崩了。
崩溃信息:

Android中非UI主线程能不能操作UI()
文章图片
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()
文章图片
所以,Android中非UI主线程能不能操作UI?答案是可以的。只不过只能在创建View的线程里操作view.
总结: 【Android中非UI主线程能不能操作UI()】1.由于Android是通过Handler消息机制的方式刷新UI的。所以Android 的UI控件是线程安全的,不会导致多线程访问使得UI处于不可预期的状态。
2.Android每次刷新UI的时候,最终根布局ViewRootImpl.checkThread()来检验线程是否是View的创建线程。

    推荐阅读