[TOC]
寫項目的過程中發現,在需要根據適當條件進行相應 UI 展示時,代碼中充斥了 setVisibility 相關的代碼,相當混亂。我們可以使用 ViewStub 來簡化相應的邏輯,并且 ViewStub 大小為 0,運行時才進行懶加載,所以性能上也有一定優勢。
# ViewStub是什么
1、ViewStub 是一個看不見的,沒有大小,不占布局位置的 View,可以用來懶加載布局。
2、當 ViewStub 變得可見或 inflate() 的時候,布局就會被加載(替換 ViewStub)。因此,ViewStub 一直存在于視圖層次結構中直到調用了 setVisibility() 或 inflate()。
3、在 ViewStub 加載完成后就會被移除,它所占用的空間就會被新的布局替換。
# 簡單使用
ViewStub 的使用很簡單,在布局文件中像引入其他控件一樣引入 ViewStub:
```xml
<ViewStub android:id="@+id/stub"
android:inflatedId="@+id/subTree"
android:layout="@layout/mySubTree"
android:layout_width="120dip"
android:layout_height="40dip" />
```
需要展示 ViewStub 所引用的視圖時,通過 id 獲取到 ViewStub:
```java
ViewStub stub = (ViewStub) findViewById(R.id.stub);
View inflated = stub.inflate();
// stub.setVisibility(View.VISIBLE);
```
除了調用 inflate 方法,還可以調用 setVisible 方法來展示 ViewStub 所引用的 View。下面,我們來看下 ViewStub 的源碼。
為統一理解,下面使用 `延遲加載 View` 來指代 ViewStub 所引用的 View。
# 源碼分析
ViewStub 這個類的代碼加上注釋也才三百多行,算是很簡單的源碼類了。先看下成員變量:
```java
/**
* 延遲加載 View 的 id
*/
private int mInflatedId;
/**
* 延遲加載 View 的布局資源
*/
private int mLayoutResource;
/**
* 延遲加載 View 對象的弱引用
*/
private WeakReference<View> mInflatedViewRef;
/**
* 布局加載器
*/
private LayoutInflater mInflater;
/**
* 延遲加載 View 加載成功回調接口
*/
private OnInflateListener mInflateListener;
```
可以通過 ViewStub 的 setOnInflateListener(OnInflateListener inflateListener) 方法設定監聽器,在視圖加載成功后執行相應的操作,其中加載成功回調接口如下:
```java
public static interface OnInflateListener {
/**
* @param stub 把加載延遲加載 View 的那個 ViewStub
* @param inflated 加載出來的那個延遲加載 View
*/
void onInflate(ViewStub stub, View inflated);
}
```
## inflate 方法
通常調用 inflate 方法來延遲加載視圖,來看看其源碼:
```java
public View inflate() {
final ViewParent viewParent = getParent();
if (viewParent != null && viewParent instanceof ViewGroup) {
if (mLayoutResource != 0) {
final ViewGroup parent = (ViewGroup) viewParent;
// 延遲加載 View inflate 出來
final View view = inflateViewNoAdd(parent);
// 把 ViewStub 自身從父布局視圖樹中刪除,延遲加載 View 加入父布局視圖樹中
replaceSelfWithView(view, parent);
mInflatedViewRef = new WeakReference<>(view);
if (mInflateListener != null) {
mInflateListener.onInflate(this, view);
}
return view;
} else {
throw new IllegalArgumentException("ViewStub must have a valid layoutResource");
}
} else {
throw new IllegalStateException("ViewStub must have a non-null ViewGroup viewParent");
}
}
```
可以看到,首先獲取到 ViewStub 的父布局并強轉為 ViewGroup,然后根據父布局的約束將延遲加載 View inflate 出來,接著把 ViewStub 自身從父布局視圖樹中刪除,延遲加載 View 加入父布局視圖樹中。
延遲加載 View 加載成功后,通過弱引用對象保存該視圖對象,并回調加載成功回調接口。至此,inflate 方法的工作也就完成了。
其中 inflateViewNoAdd 和 replaceSelfWithView 方法的源碼如下,都是很基礎的代碼:
```java
private View inflateViewNoAdd(ViewGroup parent) {
final LayoutInflater factory;
if (mInflater != null) {
factory = mInflater;
} else {
factory = LayoutInflater.from(mContext);
}
final View view = factory.inflate(mLayoutResource, parent, false);
if (mInflatedId != NO_ID) {
view.setId(mInflatedId);
}
return view;
}
private void replaceSelfWithView(View view, ViewGroup parent) {
final int index = parent.indexOfChild(this);
parent.removeViewInLayout(this);
final ViewGroup.LayoutParams layoutParams = getLayoutParams();
if (layoutParams != null) {
parent.addView(view, index, layoutParams);
} else {
parent.addView(view, index);
}
}
```
## setVisibility 方法
通過 ViewStub 的 inflate 方法,我們可以順利的把想要延遲加載的 View 加載出來了,然而實際的業務邏輯并不會這么簡單,有時可能加載出來后需要再次隱藏,隱藏后還要再次加載。或者當 inflate 方法多次調用時,就會報錯了:
```plain
java.lang.IllegalStateException: ViewStub must have a non-null ViewGroup viewParent
```
通過剛才的源碼可以看到,在調用 inflate 方法時,viewParent 為 null 了,為什么呢?原來我們在第一次調用 inflate 方法時,執行到 replaceSelfWithView 方法時,就把 ViewStub 從父布局中移除,然后把延遲加載 View 加入到父布局中了。所以現在再次調用 ViewStub 的 getParent 方法當然為 null 了。
那么延遲加載 View 是否可以多次顯示、隱藏呢,當然是可以的,接下來我們看看 ViewStub 的 setVisibility 方法。
```java
public void setVisibility(int visibility) {
if (mInflatedViewRef != null) {
View view = mInflatedViewRef.get();
if (view != null) {
view.setVisibility(visibility);
} else {
throw new IllegalStateException("setVisibility called on un-referenced view");
}
} else {
super.setVisibility(visibility);
if (visibility == VISIBLE || visibility == INVISIBLE) {
inflate();
}
}
}
```
邏輯很簡單:首先判斷延遲加載 View 的弱引用對象是否為空,不為空直接獲取該 View,并設置相應的可見性;弱引用對象為空的話,如果可見性設置為 VISIBLE 或 INVISIBLE 時,直接調用 inflate 方法進行加載。
需要注意兩點:
* 代碼中不調用 inflate 方法,直接調用 setVisibility(View.INVISIBLE) 時,會將延遲加載 View 加載顯示出來的
* ViewStub 在調用一次 inflate 方法后,延遲加載 View 的弱引用對象就不為空了,除非被垃圾回收器掃描到進行回收了。
# 繪制流程方法
```java
public ViewStub(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
//...
setVisibility(GONE);
setWillNotDraw(true);
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
setMeasuredDimension(0, 0);
}
@Override
public void draw(Canvas canvas) {
}
@Override
protected void dispatchDraw(Canvas canvas) {
}
```
可以看到,ViewStub在構造方法中調用了setWillNotDraw(true),會在后續過程中進行性能優化,略過繪制過程。在onMeasure方法中直接設置測量寬高為0,重寫了draw方法、dispatchDraw方法,方法內容為空。所以ViewStub 是一個看不見的,沒有大小,不占布局位置的 View。
# ViewStub 與 ListView 結合使用的問題
在 ListView 的 item 中使用 ViewStub 根據條件動態加載視圖,遇到視圖混亂問題,ListView 代碼片段如下:
```java
public View getView(int position, View convertView, ViewGroup parent) {
/*
if (convertView == null) {
...
} else {
...
}
...
*/
holder.mViewStub.setOnInflateListener(new ViewStub.OnInflateListener() {
@Override
public void onInflate(ViewStub stub, View inflated) {
inflated.findViewById(R.id.tv_unapply)
.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// unapply(mEntity.getJobId);
}
});
inflated.findViewById(R.id.tv_contact_company)
.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// contactCompany();
}
});
}
});
holder.mApplyViewStub.setVisibility(View.VISIBLE);
}
```
上面源碼分析時可知,ViewStub 的 setVisibility 方法在 mInflatedViewRef 不為空時,直接對延遲加載 View 進行可見性設定,并沒有執行 inflate 方法,所以我們上面設置的 onInflateListener 就沒有回調到了。
所以我們在 ViewHolder 中將 ViewStub 進行緩存后,getView 方法中再次取出時就不會調用 inflate 方法了,所以會出現 onInflateListener 方法回調異常。
# 總結
ViewStub 源碼的幾個關鍵成員變量和方法以及介紹完了。可以看到,ViewStub 的延遲加載特性,在提升視圖性能的同時,還可以使業務邏輯更清晰,大量減少 View 的 setVisibility 相關代碼。
使用時,只需要一次性加載的可以直接調用 inflate 方法進行加載。需要多次調用修改可見性的,可以調用 ViewStub 的 setVisibility 方法,同時可以通過設置監聽器在加載完成后執行相應的操作。
- 導讀
- Java知識
- Java基本程序設計結構
- 【基礎知識】Java基礎
- 【源碼分析】Okio
- 【源碼分析】深入理解i++和++i
- 【專題分析】JVM與GC
- 【面試清單】Java基本程序設計結構
- 對象與類
- 【基礎知識】對象與類
- 【專題分析】Java類加載過程
- 【面試清單】對象與類
- 泛型
- 【基礎知識】泛型
- 【面試清單】泛型
- 集合
- 【基礎知識】集合
- 【源碼分析】SparseArray
- 【面試清單】集合
- 多線程
- 【基礎知識】多線程
- 【源碼分析】ThreadPoolExecutor源碼分析
- 【專題分析】volatile關鍵字
- 【面試清單】多線程
- Java新特性
- 【專題分析】Lambda表達式
- 【專題分析】注解
- 【面試清單】Java新特性
- Effective Java筆記
- Android知識
- Activity
- 【基礎知識】Activity
- 【專題分析】運行時權限
- 【專題分析】使用Intent打開三方應用
- 【源碼分析】Activity的工作過程
- 【面試清單】Activity
- 架構組件
- 【專題分析】MVC、MVP與MVVM
- 【專題分析】數據綁定
- 【面試清單】架構組件
- 界面
- 【專題分析】自定義View
- 【專題分析】ImageView的ScaleType屬性
- 【專題分析】ConstraintLayout 使用
- 【專題分析】搞懂點九圖
- 【專題分析】Adapter
- 【源碼分析】LayoutInflater
- 【源碼分析】ViewStub
- 【源碼分析】View三大流程
- 【源碼分析】觸摸事件分發機制
- 【源碼分析】按鍵事件分發機制
- 【源碼分析】Android窗口機制
- 【面試清單】界面
- 動畫和過渡
- 【基礎知識】動畫和過渡
- 【面試清單】動畫和過渡
- 圖片和圖形
- 【專題分析】圖片加載
- 【面試清單】圖片和圖形
- 后臺任務
- 應用數據和文件
- 基于網絡的內容
- 多線程與多進程
- 【基礎知識】多線程與多進程
- 【源碼分析】Handler
- 【源碼分析】AsyncTask
- 【專題分析】Service
- 【源碼分析】Parcelable
- 【專題分析】Binder
- 【源碼分析】Messenger
- 【面試清單】多線程與多進程
- 應用優化
- 【專題分析】布局優化
- 【專題分析】繪制優化
- 【專題分析】內存優化
- 【專題分析】啟動優化
- 【專題分析】電池優化
- 【專題分析】包大小優化
- 【面試清單】應用優化
- Android新特性
- 【專題分析】狀態欄、ActionBar和導航欄
- 【專題分析】應用圖標、通知欄適配
- 【專題分析】Android新版本重要變更
- 【專題分析】唯一標識符的最佳做法
- 開源庫源碼分析
- 【源碼分析】BaseRecyclerViewAdapterHelper
- 【源碼分析】ButterKnife
- 【源碼分析】Dagger2
- 【源碼分析】EventBus3(一)
- 【源碼分析】EventBus3(二)
- 【源碼分析】Glide
- 【源碼分析】OkHttp
- 【源碼分析】Retrofit
- 其他知識
- Flutter
- 原生開發與跨平臺開發
- 整體歸納
- 狀態及狀態管理
- 零碎知識點
- 添加Flutter到現有應用
- Git知識
- Git命令
- .gitignore文件
- 設計模式
- 創建型模式
- 結構型模式
- 行為型模式
- RxJava
- 基礎
- Linux知識
- 環境變量
- Linux命令
- ADB命令
- 算法
- 常見數據結構及實現
- 數組
- 排序算法
- 鏈表
- 二叉樹
- 棧和隊列
- 算法時間復雜度
- 常見算法思想
- 其他技術
- 正則表達式
- 編碼格式
- HTTP與HTTPS
- 【面試清單】其他知識
- 開發歸納
- Android零碎問題
- 其他零碎問題
- 開發思路