Android葵花宝典基础知识Android 问题杂记

Android中非UI主线程能不能操作UI?

2018-04-22  本文已影响407人  Calllanna

问题:“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的创建线程。

上一篇 下一篇

猜你喜欢

热点阅读