佈局優化之ViewStub源碼分析

来源:https://www.cnblogs.com/ganchuanpu/archive/2018/05/23/9076744.html
-Advertisement-
Play Games

源碼分析 這是什麼玩應兒呢?其實就是一個輕量級的頁面,我們通常使用它來做預載入處理,來改善頁面載入速度和提高流暢性,ViewStub本身不會占用層級,它最終會被它指定的層級取代。 在一些場合取代android:visibility=”gone”的用法,因為被gone掉的佈局不斷是會同時創建對象的。那 ...


源碼分析

  1 @RemoteView  
  2 public final class ViewStub extends View {  
  3     private int mInflatedId;  
  4     private int mLayoutResource;  
  5   
  6     private WeakReference<View> mInflatedViewRef;  
  7   
  8     private LayoutInflater mInflater;  
  9     private OnInflateListener mInflateListener;  
 10   
 11     public ViewStub(Context context) {  
 12         this(context, 0);  
 13     }  
 14   
 15     /**  
 16      * Creates a new ViewStub with the specified layout resource.  
 17      *  
 18      * @param context The application's environment.  
 19      * @param layoutResource The reference to a layout resource that will be inflated.  
 20      */  
 21     public ViewStub(Context context, @LayoutRes int layoutResource) {  
 22         this(context, null);  
 23   
 24         mLayoutResource = layoutResource;  
 25     }  
 26   
 27     public ViewStub(Context context, AttributeSet attrs) {  
 28         this(context, attrs, 0);  
 29     }  
 30   
 31     public ViewStub(Context context, AttributeSet attrs, int defStyleAttr) {  
 32         this(context, attrs, defStyleAttr, 0);  
 33     }  
 34   
 35     public ViewStub(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {  
 36         super(context);  
 37   
 38         final TypedArray a = context.obtainStyledAttributes(attrs,  
 39                 R.styleable.ViewStub, defStyleAttr, defStyleRes);  
 40         mInflatedId = a.getResourceId(R.styleable.ViewStub_inflatedId, NO_ID);  
 41         mLayoutResource = a.getResourceId(R.styleable.ViewStub_layout, 0);  
 42         mID = a.getResourceId(R.styleable.ViewStub_id, NO_ID);  
 43         a.recycle();  
 44   
 45         setVisibility(GONE); // 預設不可見  
 46         setWillNotDraw(true); // 如果View不繪製任何內容,設置這個標記可以優化性能,預設View沒有設置這個標記,如果重寫onDraw,就不要設置這個標記  
 47     }  
 48   
 49     @Override  
 50     protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {  
 51         setMeasuredDimension(0, 0); // 測量時尺寸為0  
 52     }  
 53   
 54     @Override  
 55     public void draw(Canvas canvas) { // 不繪製內容  
 56     }  
 57   
 58     @Override  
 59     protected void dispatchDraw(Canvas canvas) {  
 60     }  
 61 ..... 省去部分代碼  
 62   
 63     private View inflateViewNoAdd(ViewGroup parent) {  
 64         final LayoutInflater factory;  
 65         if (mInflater != null) {  
 66             factory = mInflater;  
 67         } else {  
 68             factory = LayoutInflater.from(mContext);  
 69         } // 通過inflate填充佈局  
 70         final View view = factory.inflate(mLayoutResource, parent, false);  
 71   
 72         if (mInflatedId != NO_ID) {  
 73             view.setId(mInflatedId);  
 74         }  
 75         return view;  
 76     }  
 77   
 78     private void replaceSelfWithView(View view, ViewGroup parent) {  
 79         final int index = parent.indexOfChild(this);  
 80         parent.removeViewInLayout(this); // 移除ViewStub,後面不能在inflate  
 81   
 82         final ViewGroup.LayoutParams layoutParams = getLayoutParams(); // 獲得ViewStub的佈局參數  
 83         if (layoutParams != null) {  
 84             parent.addView(view, index, layoutParams); // 把ViewStub指定的佈局添加到parent中  
 85         } else {  
 86             parent.addView(view, index);  
 87         }  
 88     }  
 89   
 90     /**  
 91      * Inflates the layout resource identified by {@link #getLayoutResource()}  
 92      * and replaces this StubbedView in its parent by the inflated layout resource.  
 93      *  
 94      * @return The inflated layout resource.  
 95      *  
 96      */  
 97     public View inflate() {  
 98         final ViewParent viewParent = getParent(); // 獲取ViewStub的parent  
 99   
100         if (viewParent != null && viewParent instanceof ViewGroup) {  
101             if (mLayoutResource != 0) {  
102                 final ViewGroup parent = (ViewGroup) viewParent;  
103                 final View view = inflateViewNoAdd(parent);  
104                 replaceSelfWithView(view, parent);  
105   
106                 mInflatedViewRef = new WeakReference<>(view);  
107                 if (mInflateListener != null) {  
108                     mInflateListener.onInflate(this, view);  
109                 }  
110   
111                 return view;  
112             } else {  
113                 throw new IllegalArgumentException("ViewStub must have a valid layoutResource");  
114             }  
115         } else {  
116             throw new IllegalStateException("ViewStub must have a non-null ViewGroup viewParent");  
117         }  
118     }  
119   
120     /**  
121      * Specifies the inflate listener to be notified after this ViewStub successfully  
122      * inflated its layout resource.  
123      *  
124      * @param inflateListener The OnInflateListener to notify of successful inflation.  
125      *  
126      * @see android.view.ViewStub.OnInflateListener  
127      */  
128     public void setOnInflateListener(OnInflateListener inflateListener) {  
129         mInflateListener = inflateListener;  
130     }  
131   
132     /**  
133      * Listener used to receive a notification after a ViewStub has successfully  
134      * inflated its layout resource.  
135      *  
136      * @see android.view.ViewStub#setOnInflateListener(android.view.ViewStub.OnInflateListener)   
137      */  
138     public static interface OnInflateListener {  
139         /**  
140          * Invoked after a ViewStub successfully inflated its layout resource.  
141          * This method is invoked after the inflated view was added to the  
142          * hierarchy but before the layout pass.  
143          *  
144          * @param stub The ViewStub that initiated the inflation.  
145          * @param inflated The inflated View.  
146          */  
147         void onInflate(ViewStub stub, View inflated);  
148     }  
149   
150     /** @hide **/  
151     public class ViewReplaceRunnable implements Runnable {  
152         public final View view;  
153   
154         ViewReplaceRunnable(View view) {  
155             this.view = view;  
156         }  
157   
158         @Override  
159         public void run() {  
160             replaceSelfWithView(view, (ViewGroup) getParent());  
161         }  
162     }  
163 }  

這是什麼玩應兒呢?其實就是一個輕量級的頁面,我們通常使用它來做預載入處理,來改善頁面載入速度和提高流暢性,ViewStub本身不會占用層級,它最終會被它指定的層級取代。 
在一些場合取代android:visibility=”gone”的用法,因為被gone掉的佈局不斷是會同時創建對象的。那為什麼使用ViewStub就高效呢, 
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 
setMeasuredDimension(0, 0); 
}

@Override 
public void draw(Canvas canvas) { 

由onMeasure()方法和draw()方法可以看出, ViewStub的初始寬高都是零,所以他開始不會占用空間,其次draw()方法也沒有執行任何的繪製,由這兩個方法就可以看出,ViewStub的確很高效。 
在代碼中要操縱ViewStub的時候,要首先使用viewstub.inflate()方法,將其所擁有的View初始化進去。否則會報空指針錯誤。 
但ViewStub也不是萬能的,下麵總結下ViewStub能做的事兒和什麼時候該用ViewStub,什麼時候該用可見性的控制。

首先來說說ViewStub的一些特點:

1. ViewStub只能Inflate一次,之後ViewStub對象會被置為空。按句話說,某個被ViewStub指定的佈局被Inflate後,就不會夠再通過ViewStub來控制它了。
2. ViewStub只能用來Inflate一個佈局文件,而不是某個具體的View,當然也可以把View寫在某個佈局文件中。

基於以上的特點,那麼可以考慮使用ViewStub的情況有:
1. 在程式的運行期間,某個佈局在Inflate後,就不會有變化,除非重新啟動。
因為ViewStub只能Inflate一次,之後會被置空,所以無法指望後面接著使用ViewStub來控制佈局。所以當需要在運行時不止一次的顯示和隱藏某個佈局,那麼ViewStub是做不到的。這時就只能使用View的可見性來控制了。
2. 想要控制顯示與隱藏的是一個佈局文件,而非某個View。
因為設置給ViewStub的只能是某個佈局文件的Id,所以無法讓它來控制某個View。
所以,如果想要控制某個View(如Button或TextView)的顯示與隱藏,或者想要在運行時不斷的顯示與隱藏某個佈局或View,只能使用View的可見性來控制。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 啟動mongodb服務端 : sudo service mongod start 或者 sudo /usr/local/mongodb/bin/mongod --config /etc/mongodb.conf 停止mongodb服務端: sudo service mongodb stop 查詢mo ...
  • 1、前期準備條件,三台主機:master,slave1,slave2 2、三台機器在集群中所擔任的角色: master:namenode,datanode,nodemanager,historyserver slave1:resoucemanaer,datanode,nodemanager slav ...
  • 一、 介紹 二、 索引的原理 三、 索引的數據結構 四、 聚集索引與輔助索引 五、 MySQL索引管理 六、 測試索引 七、 正確使用索引 八、 聯合索引與覆蓋索引 九、 查詢優化神器-explain 十、 慢查詢優化的基本步驟 十一、 慢日誌管理 一 、介紹 為何要有索引? 一般的應用系統,讀寫比 ...
  • 一. 前言 對於sql server 這個產品來說,記憶體這塊是最重要的一個資源, 當我們新建一個會話,相同的sql語句查詢第二次查詢時間往往會比第一次快,特別是在sql統計或大量查詢數據輸出時,會有這麼感覺。除了第一次要編譯生成執行計劃, 在CPU,I/O 的影響外,最主要的是第二次查詢是從記憶體緩存 ...
  • 最近一個項目要部署在阿裡雲上,為了開發團隊方便,我自費買了個ECS,先裝個資料庫給開發用。 因為之前都是在真機安裝,與這次阿裡雲上的部署比起來,還是有點區別的。 Mysql 1 安裝mysql版本包 wget https://dev.mysql.com/get/mysql57-community-r ...
  • 情景:利用Dataguard滾動方式升級資料庫後,備庫應用redo報錯:ORA-10485 MRP0: Background Media Recovery terminated with error 10485Errors in file /home/u01/app/diag/rdbms/orcl1 ...
  • 在網路上直接看電影已經不是什麼新鮮的事情,在iOS等移動設備上也有很多線上視頻應用,如國內的PPS和PPLive應用,還有一些新聞視頻都可以線上觀看,如USA TODY。所以這些線上視頻都採用流媒體技術。 網路多媒體文件播放受到網路帶寬,文件格式,文件大小等因素的影響,情況變得非常的複雜。為了提高速 ...
  • 無論是調試Web頁面還是調試Hybrid混合應用,只要是調試Android的webview,都需要使用Chrome://inspect進行調試。但是國內開發者會出現404 Not Found錯誤: 解決方法有兩種: ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...