亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > Android > 正文

Android子線程與更新UI問題的深入講解

2019-10-21 21:31:02
字體:
來源:轉載
供稿:網友

前言

在Android項目中經常有碰到這樣的問題,在子線程中完成耗時操作之后要更新UI,下面就自己經歷的一些項目總結一下更新的方法。話不多說了,來一起看看詳細的介紹吧

引子:

情形1

 @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); TextView textView = findViewById(R.id.home_tv); ImageView imageView = findViewById(R.id.home_img); new Thread(new Runnable() {  @Override  public void run() {  textView.setText("更新TextView");  imageView.setImageResource(R.drawable.img);  } }).start(); }

運行結果:正常運行!??!

情形二

 @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); TextView textView = findViewById(R.id.home_tv); ImageView imageView = findViewById(R.id.home_img); new Thread(new Runnable() {  @Override  public void run() {  try {   Thread.sleep(5000);  } catch (InterruptedException e) {   e.printStackTrace();  }  textView.setText("更新TextView");  imageView.setImageResource(R.drawable.img);  } }).start(); }

運行結果:異常

    android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
        at android.view.ViewRootImpl.checkThread(ViewRootImpl.java:6357)
        at android.view.ViewRootImpl.requestLayout(ViewRootImpl.java:874)
        at android.view.View.requestLayout(View.java:17476)
        at android.view.View.requestLayout(View.java:17476)
        at android.view.View.requestLayout(View.java:17476)
        at android.view.View.requestLayout(View.java:17476)
        at android.view.View.requestLayout(View.java:17476)
        at android.view.View.requestLayout(View.java:17476)
        at android.widget.RelativeLayout.requestLayout(RelativeLayout.java:360)
        at android.view.View.requestLayout(View.java:17476)
        at android.widget.TextView.checkForRelayout(TextView.java:6871)
        at android.widget.TextView.setText(TextView.java:4057)
        at android.widget.TextView.setText(TextView.java:3915)
        at android.widget.TextView.setText(TextView.java:3890)
        at com.dong.demo.MainActivity$1.run(MainActivity.java:44)
        at java.lang.Thread.run(Thread.java:818)

不是說,子線程不能更新UI嗎,為什么情形一可以正常運行,情形二不能正常運行呢;

子線程修改UI出現異常,與什么方法有關

首先從出現異常的log日志入手,發現出現異常的方法調用順序如下:

TextView.setText(TextView.java:4057)

TextView.checkForRelayout(TextView.java:6871)

View.requestLayout(View.java:17476)

RelativeLayout.requestLayout(RelativeLayout.java:360)

View.requestLayout(View.java:17476)

ViewRootImpl.requestLayout(ViewRootImpl.java:874)

ViewRootImpl.checkThread(ViewRootImpl.java:6357)

更改ImageView時,出現的異常類似;

首先看TextView.setText()方法的源碼

 private void setText(CharSequence text, BufferType type,    boolean notifyBefore, int oldlen) {  //省略其他代碼 if (mLayout != null) {  checkForRelayout(); } sendOnTextChanged(text, 0, oldlen, textLength); onTextChanged(text, 0, oldlen, textLength); //省略其他代碼

然后,查看以下checkForRelayout()方法的與源碼。

 private void checkForRelayout() { // If we have a fixed width, we can just swap in a new text layout // if the text height stays the same or if the view height is fixed. if ((mLayoutParams.width != LayoutParams.WRAP_CONTENT  //省略代碼  // We lose: the height has changed and we have a dynamic height.  // Request a new view layout using our new text layout.  requestLayout();  invalidate(); } else {  // Dynamic width, so we have no choice but to request a new  // view layout with a new text layout.  nullLayouts();  requestLayout();  invalidate(); } }

checkForReLayout方法,首先會調用需要改變的View的requestLayout方法,然后執行invalidate()重繪操作;

TextView沒有重寫requestLayout方法,requestLayout方法由View實現;

查看RequestLayout方法的源碼:

 public void requestLayout() { //省略其他代碼 if (mParent != null && !mParent.isLayoutRequested()) {  mParent.requestLayout(); } if (mAttachInfo != null && mAttachInfo.mViewRequestingLayout == this) {  mAttachInfo.mViewRequestingLayout = null; } }

View獲取到父View(類型是ViewParent,ViewPaerent是個接口,requestLayout由子類來具體實現),mParent,然后調用父View的requestLayout方法,比如示例中的父View就是xml文件的根布局就是RelativeLayout。

 @Override public void requestLayout() { super.requestLayout(); mDirtyHierarchy = true; }

繼續跟蹤super.requestLayout()方法,即ViewGroup沒有重新,即調用的是View的requestLayout方法。

經過一系列的調用ViewParent的requestLayout方法,最終調用到ViewRootImp的requestLayout方法。ViewRootImp實現了ViewParent接口,繼續查看ViewRootImp的requestLayout方法源碼。

 @Override public void requestLayout() {  if (!mHandlingLayoutInLayoutRequest) {   checkThread();   mLayoutRequested = true;   scheduleTraversals();  } }

ViewRootImp的requestLayout方法中有兩個方法:

一、checkThread,檢查線程,源碼如下

 void checkThread() {  if (mThread != Thread.currentThread()) {   throw new CalledFromWrongThreadException(     "Only the original thread that created a view hierarchy can touch its views.");  } }

判斷當前線程,是否是創建ViewRootImp的線程,而創建ViewRootImp的線程就是主線程,當前線程不是主線程的時候,就拋出異常。

二、scheduleTraversals(),查看源碼:

 void scheduleTraversals() {  if (!mTraversalScheduled) {   mTraversalScheduled = true;   mTraversalBarrier = mHandler.getLooper().getQueue().postSyncBarrier();   mChoreographer.postCallback(     Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);   if (!mUnbufferedInputDispatch) {    scheduleConsumeBatchedInput();   }   notifyRendererOfFramePending();   pokeDrawLockIfNeeded();  } }

查看mTraversalRunnable中run()方法的具體操作

 final class TraversalRunnable implements Runnable {  @Override  public void run() {   doTraversal();  } }

繼續追蹤doTraversal()方法

 void doTraversal() {  if (mTraversalScheduled) {   mTraversalScheduled = false;   mHandler.getLooper().getQueue().removeSyncBarrier(mTraversalBarrier);   if (mProfile) {    Debug.startMethodTracing("ViewAncestor");   }   performTraversals();   if (mProfile) {    Debug.stopMethodTracing();    mProfile = false;   }  } }

查看到performTraversals()方法,熟悉了吧,這是View繪制的起點。

Android,子線程,更新UI

總結一下:

1.Android更新UI會調用View的requestLayout()方法,在requestLayout方法中,獲取ViewParent,然后調用ViewParent的requestLayout()方法,一直調用下去,直到調用到ViewRootImp的requestLayout方法;

2.ViewRootImp的requetLayout方法,主要有兩部操作一個是checkThread()方法,檢測線程,一個是scheduleTraversals,執行繪制相關工作;

情形3

 @Override protected void onCreate(Bundle savedInstanceState) {  Log.i("Dong", "Activity: onCreate");  super.onCreate(savedInstanceState);  setContentView(R.layout.activity_main);  new Thread(new Runnable() {   @Override   public void run() {    Looper.prepare();    try {     Thread.sleep(5000);    } catch (InterruptedException e) {     e.printStackTrace();    }    Toast.makeText(MainActivity.this, "顯示Toast", Toast.LENGTH_LONG).show();    Looper.loop();   }  }).start(); }

運行結果:正常

分析

下面從Toast源碼進行分析:

 public static Toast makeText(Context context, CharSequence text, @Duration int duration) {  return makeText(context, null, text, duration); }

makeText方法調用了他的重載方法,繼續追蹤

 public static Toast makeText(@NonNull Context context, @Nullable Looper looper,   @NonNull CharSequence text, @Duration int duration) {  Toast result = new Toast(context, looper);  LayoutInflater inflate = (LayoutInflater)    context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);  View v = inflate.inflate(com.android.internal.R.layout.transient_notification, null);  TextView tv = (TextView)v.findViewById(com.android.internal.R.id.message);  tv.setText(text);  result.mNextView = v;  result.mDuration = duration;  return result; }

新建了一個Toast對象,然后對顯示的布局、內容、時長進行了設置,并返回Toast對象。

繼續查看new Toast()的源碼

 public Toast(@NonNull Context context, @Nullable Looper looper) {  mContext = context;  mTN = new TN(context.getPackageName(), looper);  mTN.mY = context.getResources().getDimensionPixelSize(    com.android.internal.R.dimen.toast_y_offset);  mTN.mGravity = context.getResources().getInteger(    com.android.internal.R.integer.config_toastDefaultGravity); }

繼續查看核心代碼 mTN = new TN(context.getPackageName(), looper);

TN初始化的源碼為:

  TN(String packageName, @Nullable Looper looper) {   //省略部分不相關代碼   if (looper == null) {    // 沒有傳入Looper對象的話,使用當前線程對應的Looper對象    looper = Looper.myLooper();    if (looper == null) {     throw new RuntimeException(       "Can't toast on a thread that has not called Looper.prepare()");    }   }   //初始化了Handler對象   mHandler = new Handler(looper, null) {    @Override    public void handleMessage(Message msg) {     switch (msg.what) {      case SHOW: {       IBinder token = (IBinder) msg.obj;       handleShow(token);       break;      }      case HIDE: {       handleHide();       // Don't do this in handleHide() because it is also invoked by       // handleShow()       mNextView = null;       break;      }      case CANCEL: {       handleHide();       // Don't do this in handleHide() because it is also invoked by       // handleShow()       mNextView = null;       try {        getService().cancelToast(mPackageName, TN.this);       } catch (RemoteException e) {       }       break;      }     }    }   };  }

繼續追蹤handleShow(token)方法:

  public void handleShow(IBinder windowToken) {   //省略部分代碼   if (mView != mNextView) {    // remove the old view if necessary    handleHide();    mView = mNextView;    Context context = mView.getContext().getApplicationContext();    String packageName = mView.getContext().getOpPackageName();    if (context == null) {     context = mView.getContext();    }    mWM = (WindowManager)context.getSystemService(Context.WINDOW_SERVICE);    /*    ·*省略設置顯示屬性的代碼    ·*/    if (mView.getParent() != null) {     if (localLOGV) Log.v(TAG, "REMOVE! " + mView + " in " + this);     mWM.removeView(mView);    }=    try {     mWM.addView(mView, mParams);     trySendAccessibilityEvent();    } catch (WindowManager.BadTokenException e) {     /* ignore */    }   }  }

通過源碼可以看出,Toast顯示內容是通過mWM(WindowManager類型)的直接添加的,更正:mWm.addView 時,對應的ViewRootImp初始化發生在子線程,checkThread方法中的mThread != Thread.currentThread()判斷為true,所以不會拋出只能在主線程更新UI的異常。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對VEVB武林網的支持。


注:相關教程知識閱讀請移步到Android開發頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲男人的天堂网站| 亚洲一区二区三区777| 亚洲精品久久久久久久久久久久| 成人黄色免费在线观看| 欧日韩在线观看| 亚洲第一av网| 日韩在线国产精品| 日韩欧美主播在线| 成人亚洲激情网| 国内精品美女av在线播放| 亚洲精品一区二区三区婷婷月| 青青草精品毛片| 91在线精品播放| 久久艹在线视频| 亚洲电影免费在线观看| 成人黄色免费看| 欧美激情奇米色| 国产欧美一区二区白浆黑人| 亚洲精品美女在线| 亚洲一区国产精品| 日韩视频永久免费观看| 91网站在线看| 国产精品免费视频久久久| 丝袜亚洲另类欧美重口| www.亚洲免费视频| 九九九久久久久久| 久久成人精品一区二区三区| 亚洲欧美色婷婷| 国产成人拍精品视频午夜网站| 97国产精品人人爽人人做| 国内精品视频久久| 亚洲va欧美va国产综合久久| 久久久久久久电影一区| 神马久久久久久| 欧美视频精品一区| 国产69精品久久久久久| 国产精品视频色| 国产精品美女无圣光视频| 亚洲综合一区二区不卡| 日韩一区视频在线| 91国偷自产一区二区三区的观看方式| 亚洲免费高清视频| 国产精品video| 久久免费精品视频| 中文字幕综合一区| 久久久www成人免费精品张筱雨| 欧美电影免费看| 亚洲一区二区三区四区在线播放| 91极品视频在线| 久久影院免费观看| 亚洲欧美一区二区三区久久| 久久亚洲国产成人| 欧美亚洲国产另类| 欧洲一区二区视频| 最近2019中文字幕第三页视频| 欧美日韩国产黄| 亚洲最新在线视频| 久久久久久国产| 亚洲女同性videos| 69av视频在线播放| 亚洲综合在线中文字幕| 久久精彩免费视频| 久久久成人的性感天堂| 精品亚洲一区二区三区在线播放| 久久九九有精品国产23| 国产成人精品久久二区二区91| 国产欧洲精品视频| 欧美怡春院一区二区三区| 日韩免费观看av| 亚洲人成电影网站色| 成人h猎奇视频网站| 亚洲第一免费播放区| 欧美激情免费观看| 亚洲国产欧美一区二区三区久久| 久久久久久久久久国产| 91九色在线视频| 国产一区二区激情| 成人伊人精品色xxxx视频| 国产精品永久免费视频| 国产一区二区三区网站| 日本久久精品视频| 日韩一区二区精品视频| 成人信息集中地欧美| 成人中心免费视频| 日韩成人久久久| 国产精品三级在线| 国产午夜精品麻豆| 欧美午夜电影在线| 黑人精品xxx一区| 69影院欧美专区视频| 97在线视频精品| 97国产精品视频人人做人人爱| 国产日韩欧美在线播放| xxx一区二区| 亚洲电影在线看| 国产精品∨欧美精品v日韩精品| 日本亚洲欧美成人| 亚洲香蕉av在线一区二区三区| 一个人看的www欧美| 欧美日韩一区二区在线播放| 亚洲国产一区自拍| 亚洲奶大毛多的老太婆| 国产免费一区二区三区香蕉精| 国产精品白嫩美女在线观看| 色偷偷av亚洲男人的天堂| 亚洲国产精品系列| 亚洲一区二区三区四区在线播放| 欧美成年人在线观看| 亚洲精品视频在线观看视频| 俺去了亚洲欧美日韩| 国产精品久久久久久久久久免费| 黑丝美女久久久| 欧美壮男野外gaytube| 精品中文字幕在线2019| 欧美日韩国产精品一区二区不卡中文| 国产精品免费在线免费| 精品亚洲aⅴ在线观看| 少妇av一区二区三区| 欧美激情影音先锋| 欧美黄色片免费观看| 尤物tv国产一区| 97人洗澡人人免费公开视频碰碰碰| 国产亚洲一区二区在线| 深夜福利日韩在线看| 日韩在线观看免费av| 国产精品自产拍在线观看| 91日韩在线视频| 狠狠做深爱婷婷久久综合一区| 亚洲男人天堂九九视频| 丝袜亚洲另类欧美重口| 日韩一中文字幕| 色av吧综合网| 亚洲女人天堂色在线7777| 91极品视频在线| 欧美电影免费观看大全| 亚洲综合小说区| 国产成人一区二区三区电影| 日韩一区av在线| 国产成人久久精品| xxxx欧美18另类的高清| 成人久久久久久久| 日韩av手机在线看| 理论片在线不卡免费观看| 久久国产精品免费视频| 正在播放欧美一区| 国产精品海角社区在线观看| 亚洲色图av在线| 亚洲精品丝袜日韩| 欧美日韩亚洲视频一区| 欧美裸体男粗大视频在线观看| 91丨九色丨国产在线| 日韩美女福利视频| 亚洲免费人成在线视频观看| 色黄久久久久久| 欧美裸体男粗大视频在线观看| 91国产视频在线播放| 久久久久久久影视| 色悠悠国产精品| 亚洲性无码av在线| 精品日本美女福利在线观看| 亚洲人成在线一二| 91经典在线视频| 俺也去精品视频在线观看| 国产亚洲精品91在线|