成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Android進階之源碼中分析View.post()為何獲取控件寬高

開發(fā) 前端
為什么 View.post() 的操作是可以對 UI 進行操作的呢,即使是在子線程中調用 View.post()? 今天我們就來分析分析

[[425705]]

本文轉載自微信公眾號「Android開發(fā)編程」,作者Android開發(fā)編程。轉載本文請聯(lián)系Android開發(fā)編程公眾號。

前言

為什么 View.post() 的操作是可以對 UI 進行操作的呢,即使是在子線程中調用 View.post()?

今天我們就來分析分析

一、View.post源碼深入分析

1、View.post()

View 的 post 方法如下:

  1. public boolean post(Runnable action) { 
  2.     // 1、首先判斷AttachInfo是否為null 
  3.     final AttachInfo attachInfo = mAttachInfo; 
  4.     if (attachInfo != null) { 
  5.         // 1.1如果不為null,直接調用其內部Handler的post 
  6.         return attachInfo.mHandler.post(action); 
  7.     } 
  8.     // 2、否則加入當前View的等待隊列 
  9.     getRunQueue().post(action); 
  10.     return true
  • AttachInfo 是 View 的靜態(tài)內部類,每個 View 都會持有一個 AttachInfo,它默認為 null;
  • 如果mAttachInfo為空,就執(zhí)行:把action加入當前view的等待隊列;

2、getRunQueue().post()

看下 getRunQueue().post():

  1. private HandlerActionQueue getRunQueue() { 
  2.     if (mRunQueue == null) { 
  3.         mRunQueue = new HandlerActionQueue(); 
  4.     } 
  5.     return mRunQueue; 
  • getRunQueue() 返回的是 HandlerActionQueue;
  • 調用了 HandlerActionQueue 的 post 方法:
  1. public void post(Runnable action) { 
  2.     // 調用到postDelayed方法,這有點類似于Handler發(fā)送消息 
  3.     postDelayed(action, 0); 
  1. // 實際調用postDelayed 
  2. public void postDelayed(Runnable action, long delayMillis) { 
  3.     // HandlerAction表示要執(zhí)行的任務 
  4.     final HandlerAction handlerAction = new HandlerAction(action, delayMillis); 
  5.     synchronized (this) { 
  6.         if (mActions == null) { 
  7.             // 創(chuàng)建一個保存HandlerAction的數組 
  8.             mActions = new HandlerAction[4]; 
  9.         } 
  10.         // 表示要執(zhí)行的任務HandlerAction 保存在 mActions 數組中 
  11.         mActions = GrowingArrayUtils.append(mActions, mCount, handlerAction); 
  12.         // mActions數組下標位置累加1 
  13.         mCount++; 
  14.     } 

3、HandlerAction

HandlerAction 表示一個待執(zhí)行的任務,內部持有要執(zhí)行的 Runnable 和延遲時間;=

  1. private static class HandlerAction { 
  2.     // post的任務 
  3.     final Runnable action
  4.     // 延遲時間 
  5.     final long delay; 
  6.     public HandlerAction(Runnable action, long delay) { 
  7.         this.action = action
  8.         this.delay = delay; 
  9.     } 
  10.     // 比較是否是同一個任務 
  11.     // 用于匹配某個 Runnable 和對應的HandlerAction 
  12.     public boolean matches(Runnable otherAction) { 
  13.         return otherAction == null && action == null 
  14.                 || action != null && action.equals(otherAction); 
  15.     } 

postDelayed() 創(chuàng)建一個默認長度為 4 的 HandlerAction 數組,用于保存 post() 添加的任務;

梳理總結:

  • 我們調用 View.post(Runnable) 傳進去的 Runnable 操作,在傳到 HandlerActionQueue 里會先經過 HandlerAction 包裝一下,然后再緩存起來;
  • 在 執(zhí)行 View.post(Runnable) 時,因為這時候 View 還沒有 attachedToWindow,所以這些 Runnable 操作其實并沒有被執(zhí)行,而是先通過 HandlerActionQueue 緩存起來;

4、AttachInfo

看下 AttachInfo 的創(chuàng)建過程,先看下它的構造方法:

  1. AttachInfo(IWindowSession session, IWindow window, Display display, 
  2.                ViewRootImpl viewRootImpl, Handler handler, Callbacks effectPlayer, 
  3.                Context context) { 
  4.         mSession = session; 
  5.         mWindow = window; 
  6.         mWindowToken = window.asBinder(); 
  7.         mDisplay = display; 
  8.         // 持有當前ViewRootImpl 
  9.         mViewRootImpl = viewRootImpl; 
  10.         // 當前渲染線程Handler 
  11.         mHandler = handler; 
  12.         mRootCallbacks = effectPlayer; 
  13.         mTreeObserver = new ViewTreeObserver(context); 
  14.     } 

AttachInfo 中持有當前線程的 Handler;

4.1、mAttachInfo 賦值:

  1. void dispatchAttachedToWindow(AttachInfo info, int visibility) { 
  2.     // 給當前View賦值AttachInfo,此時所有的View共用同一個AttachInfo(同一個ViewRootImpl內) 
  3.     mAttachInfo = info; 
  4.     if (mOverlay != null) { 
  5.         // 任何一個View都有一個ViewOverlay 
  6.         // ViewGroup的是ViewGroupOverlay 
  7.         // 它區(qū)別于直接在類似RelativeLaout/FrameLayout添加View,通過ViewOverlay添加的元素沒有任何事件 
  8.         // 此時主要分發(fā)給這些View浮層 
  9.         mOverlay.getOverlayView().dispatchAttachedToWindow(info, visibility); 
  10.     } 
  11.     mWindowAttachCount++; 
  12.     if ((mPrivateFlags & PFLAG_SCROLL_CONTAINER) != 0) { 
  13.         mAttachInfo.mScrollContainers.add(this); 
  14.         mPrivateFlags |= PFLAG_SCROLL_CONTAINER_ADDED; 
  15.     } 
  16.     //  mRunQueue,就是在前面的 getRunQueue().post() 
  17.     // 實際類型是 HandlerActionQueue,內部保存了當前View.post的任務 
  18.     if (mRunQueue != null) { 
  19.         // 執(zhí)行使用View.post的任務 
  20.         // 注意這里是post到渲染線程的Handler中 
  21.         mRunQueue.executeActions(info.mHandler); 
  22.         // 保存延遲任務的隊列被置為null,因為此時所有的View共用AttachInfo 
  23.         mRunQueue = null
  24.     } 
  25.     performCollectViewAttributes(mAttachInfo, visibility); 
  26.     // 回調View的onAttachedToWindow方法 
  27.     // 在Activity的onResume方法中調用,但是在View繪制流程之前 
  28.     onAttachedToWindow(); 
  29.     ListenerInfo li = mListenerInfo; 
  30.     final CopyOnWriteArrayList<OnAttachStateChangeListener> listeners = 
  31.             li != null ? li.mOnAttachStateChangeListeners : null
  32.     if (listeners != null && listeners.size() > 0) { 
  33.         for (OnAttachStateChangeListener listener : listeners) { 
  34.             // 通知所有監(jiān)聽View已經onAttachToWindow的客戶端,即view.addOnAttachStateChangeListener(); 
  35.             // 但此時View還沒有開始繪制,不能正確獲取測量大小或View實際大小 
  36.             listener.onViewAttachedToWindow(this); 
  37.         } 
  38.     } 

5、executeActions

mRunQueue 就是保存了 View.post() 任務的 HandlerActionQueue;此時調用它的 executeActions 方法如下:

  1. public void executeActions(Handler handler) { 
  2.     synchronized (this) { 
  3.         // 任務隊列 
  4.         final HandlerAction[] actions = mActions; 
  5.         // 遍歷所有任務 
  6.         for (int i = 0, count = mCount; i < count; i++) { 
  7.             final HandlerAction handlerAction = actions[i]; 
  8.             //發(fā)送到Handler中,等待執(zhí)行 
  9.             handler.postDelayed(handlerAction.action, handlerAction.delay); 
  10.         } 
  11.         //此時不在需要,后續(xù)的post,將被添加到AttachInfo中 
  12.         mActions = null
  13.         mCount = 0; 
  14.     } 
  • 遍歷所有已保存的任務,發(fā)送到 Handler 中排隊執(zhí)行;
  • 將保存任務的 mActions 置為 null,因為后續(xù) View.post() 直接添加到 AttachInfo 內部的 Handler;

6、performTraversals

  • dispatchAttachedToWindow() 的調用時機是在 View 繪制流程的開始階段;
  • 在 ViewRootImpl 的 performTraversals 方法,在該方法將會依次完成 View 繪制流程的三大階段:測量、布局和繪制;
  1. // View 繪制流程開始在 ViewRootImpl 
  2. private void performTraversals() { 
  3.     // mView是DecorView 
  4.     final View host = mView; 
  5.     if (mFirst) { 
  6.         ..... 
  7.         // host為DecorView 
  8.         // 調用DecorVIew 的 dispatchAttachedToWindow,并且把 mAttachInfo 給子view 
  9.         host.dispatchAttachedToWindow(mAttachInfo, 0); 
  10.         mAttachInfo.mTreeObserver.dispatchOnWindowAttachedChange(true); 
  11.         dispatchApplyInsets(host); 
  12.         ..... 
  13.     }  
  14.    mFirst=false 
  15.    getRunQueue().executeActions(mAttachInfo.mHandler); 
  16.    // View 繪制流程的測量階段 
  17.    performMeasure(); 
  18.    // View 繪制流程的布局階段 
  19.    performLayout(); 
  20.    // View 繪制流程的繪制階段 
  21.    performDraw(); 

7、dispatchAttachedToWindow

  • 每個 Activity 都有一個關聯(lián)的 Window 對象,用來描述應用程序窗口,每個窗口內部又包含一個 DecorView 對象,DecorView 對象用來描述窗口的視圖 — xml 布局;
  • 通過 setContentView() 設置的 View 布局最終添加到 DecorView 的 content 容器中;
  • DecorView 的 dispatchAttachedToWindow 方法的執(zhí)行過程,DecorView 并沒有重寫該方法,而是在其父類 ViewGroup 中:
  1. void dispatchAttachedToWindow(AttachInfo info, int visibility) { 
  2.     mGroupFlags |= FLAG_PREVENT_DISPATCH_ATTACHED_TO_WINDOW; 
  3.     super.dispatchAttachedToWindow(info, visibility); 
  4.     mGroupFlags &= ~FLAG_PREVENT_DISPATCH_ATTACHED_TO_WINDOW; 
  5.     // 子View的數量 
  6.     final int count = mChildrenCount; 
  7.     final View[] children = mChildren; 
  8.     // 遍歷所有子View 
  9.     for (int i = 0; i < count; i++) { 
  10.         final View child = children[i]; 
  11.         // 遍歷調用所有子View的dispatchAttachedToWindow 
  12.         // 為每個子View關聯(lián)AttachInfo 
  13.         child.dispatchAttachedToWindow(info, 
  14.                 combineVisibility(visibility, child.getVisibility())); 
  15.     } 
  • for 循環(huán)遍歷當前 ViewGroup 的所有 childView,為其關聯(lián) AttachInfo;
  • 子 View 的 dispatchAttachedToWindow,首先為當前 View 關聯(lián) AttachInfo,然后將之前 View.post() 保存的任務添加到 AttachInfo 內部的 Handler;
  • performTraversals() 會先執(zhí)行 dispatchAttachedToWindow(),這時候所有子 View 通過 View.post(Runnable) 緩存起來的 Runnable 操作就都會通過 mAttachInfo.mHandler 的 post() 方法將這些 Runnable 封裝到 Message 里發(fā)送到 MessageQueue 里;
  • mHandler 我們上面也分析過了,綁定的是主線程的 Looper,所以這些 Runnable 其實都是發(fā)送到主線程的 MessageQueue 里排隊,等待執(zhí)行。然后 performTraversals() 繼續(xù)往下工作,相繼執(zhí)行 performMeasure(),performLayout() 等操作;
  • 等全部執(zhí)行完后,表示這個 Message 已經處理完畢,所以 Looper 才會去取下一個 Message,這時候,才有可能輪到這些 Runnable 執(zhí)行;
  • 所以,這些 Runnable 操作也就肯定會在 performMeasure() 操作之后才執(zhí)行,寬高也就可以獲取到了;

總結

View.post(Runnable) 內部會自動分兩種情況處理,當 View 還沒 attachedToWindow 時,會先將這些 Runnable 操作緩存下來;否則就直接通過 mAttachInfo.mHandler 將這些 Runnable 操作 post 到主線程的 MessageQueue 中等待執(zhí)行;

View.post() 任務能夠保證在所有 View 繪制流程結束之后被調用,故如果需要依賴 View 繪制任務,此時可以優(yōu)先考慮使用該機制;

 

責任編輯:武曉燕 來源: Android開發(fā)編程
相關推薦

2021-05-17 09:50:06

Kubebuilde源碼CURD

2024-06-19 08:32:13

2017-09-05 15:27:33

View Api23Api24

2017-04-28 09:58:21

AndroidLinearLayou寬高

2016-09-22 15:50:38

JavascriptRedux源碼解析

2021-08-17 13:41:11

AndroidView事件

2019-12-23 09:13:11

Python數據語言

2025-06-26 08:24:11

AndroidView尺寸消失術

2021-12-06 14:52:08

動畫Android補間動畫

2016-03-14 09:43:47

androidview總結

2021-09-30 07:36:51

AndroidViewDraw

2021-09-05 07:35:58

lifecycleAndroid組件原理

2011-03-23 10:30:01

LAMPApache源碼

2021-10-03 15:08:32

Android

2021-09-02 07:00:01

Glide流程Android

2021-08-28 07:48:04

AndroidActivityRecActivitySta

2022-04-06 14:55:45

Harmony同步機制鴻蒙

2021-09-09 06:55:43

AndroidViewDragHel原理

2021-09-07 06:40:25

AndroidLiveData原理

2010-02-06 13:28:31

Android源碼
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 精品免费国产一区二区三区四区介绍 | 亚洲精品乱码久久久久久蜜桃 | 国产激情视频在线 | 日韩三级一区 | 免费看国产一级特黄aaaa大片 | 天天射网站| 久久久女女女女999久久 | 成人国产精品久久 | 欧美精品久久久 | 国产在线观看 | 黄色免费av | 国产一区二区三区 | 国产精品久久久久久久免费大片 | 精品久久久久久久久久久 | 欧美激情在线精品一区二区三区 | 2018天天干天天操 | www.久| 中文字幕加勒比 | 国产成人99久久亚洲综合精品 | 成年人视频在线免费观看 | 亚洲情侣视频 | 国产日韩一区二区 | 亚洲国产精品久久 | 99热首页| 亚洲小视频在线观看 | 久久久久久久成人 | 午夜视频在线观看一区二区 | 日本超碰| 中文字幕 亚洲一区 | 久久精品一 | 久久久999精品 | av片网| www.成人.com| 中文字幕成人在线 | 久久国产婷婷国产香蕉 | 久久这里只有精品首页 | 第一福利社区1024 | 桃色五月| 亚洲国产精品区 | 国产精品久久久久久久久久免费看 | 亚洲精品国产成人 |