美文网首页Android葵花宝典基础知识Android 问题杂记
Android中非UI主线程能不能操作UI?

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

作者: Calllanna | 来源:发表于2018-04-22 21:45 被阅读407次

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

    相关文章

      网友评论

      本文标题:Android中非UI主线程能不能操作UI?

      本文链接:https://www.haomeiwen.com/subject/fmhhlftx.html