首頁 > 軟體

Android ViewStub使用方法學習

2022-11-13 14:00:22

前言

當渲染一個活動時,這個活動的佈局可能會有很多visible為invisible和gone的情況,雖然這些控制元件雖然現在不顯示在螢幕上,但是系統在載入這個佈局檔案時還是會載入它的,這就影響了這個頁面的載入效率,因為這些不可見的控制元件提前載入它們並沒有什麼實際的意義,反而會減緩頁面的載入時間,所以為了解決這個問題可以使用ViewStub來懶載入暫時不顯示的佈局.

1.ViewStub的優勢

簡單來說, ViewStub可以做到按需載入一個佈局,我們可以控制它載入的時機,而不是在Activity的onCreate方法中去載入.即懶載入

2.ViewStub的使用

    <ViewStub
        android:id="@+id/stub"
        android:inflatedId="@+id/text"
        android:layout="@layout/text_view_stub"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        app:layout_constraintTop_toBottomOf="@id/textView3"
        android:layout_marginTop="180dp"
        android:layout_marginLeft="100dp"/>

屬性 功能

android:inflatedId="@+id/text" 為我們要載入的佈局提供一個id android:layout 我們需要載入的佈局 除此之外

        app:layout_constraintTop_toBottomOf="@id/textView3"
        android:layout_marginTop="180dp"
        android:layout_marginLeft="100dp"/>

這些代表我們懶載入的佈局在父佈局的位置,如果懶載入的佈局有相同的屬性,將會被覆蓋

//通過id得到viewStub物件
ViewStub viewStub = findViewById(R.id.stub);
//動態載入佈局
 viewStub.inflate();

簡單實戰

1.viewstub就是動態載入試圖

也就是在我們的app啟動繪製頁面的時候,他不會繪製到view樹中;當在程式碼中執行inflate操作後,她才會被新增到試圖中。其實ViewStub就是一個寬高都為0的一個View,它預設是不可見的,只有通過呼叫setVisibility函數或者Inflate函數才 會將其要裝載的目標佈局給載入出來,從而達到延遲載入的效果,這個要被載入的佈局通過android:layout屬性來設定。最終目的是把app載入頁面的速度提高了,使使用者體驗更好。

2.看一個簡單的demo

viewstub.xml

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:id="@+id/inflatedStart"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <TextView
        android:id="@+id/hello_tv"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toRightOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        android:text="DATA EMPTY!"/>
</android.support.constraint.ConstraintLayout>

activity_myviewstub.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <Button
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:onClick="inflate"
        android:text="inflate" />
    <Button
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:onClick="setData"
        android:text="setdata"/>
    <Button
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:onClick="hide"
        android:text="hide"/>
    <Button
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:onClick="show"
        android:text="show"/>
    <ViewStub
        android:id="@+id/vs"
        android:inflatedId="@+id/inflatedStart"
        android:layout="@layout/viewstub"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />
</LinearLayout>

MyViewStubActivity.java

package com.ysl.myandroidbase.viewstub;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.constraint.ConstraintLayout;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.view.ViewStub;
import android.widget.TextView;
import com.ysl.myandroidbase.R;
public class MyViewStubActivity extends AppCompatActivity {
    private ViewStub viewStub;
    private TextView textView;
    private View inflate;
    private ConstraintLayout constraintLayout;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_myviewstub);
        viewStub = findViewById(R.id.vs);
        //textView  = (TextView) findViewById(R.id.hello_tv);空指標,因為viewstub沒有inflate
    }
    public  void inflate(View view){
        if (inflate == null) {//inflate只會進行一次,當第二次呼叫的時候,就會拋異常;也可以try catch進行處理
            inflate = viewStub.inflate();
            constraintLayout = findViewById(R.id.inflatedStart);
            System.out.println(constraintLayout);
            System.out.println("viewStub-------->"+viewStub);
            textView  = viewStub.findViewById(R.id.hello_tv);//獲取到的textview是空的;
            System.out.println("viewStub textView-------->"+textView);//null
            textView  = constraintLayout.findViewById(R.id.hello_tv);
            System.out.println("constraintLayout textView-------->"+textView);
            textView  = findViewById(R.id.hello_tv);
            System.out.println("textView-------->"+textView);
        }
    }
    public void setData(View view){
        if (constraintLayout != null) {
            textView = constraintLayout.findViewById(R.id.hello_tv);
            textView.setText("HAVE DATA !!!");
        }
    }
    public void hide(View view){
        viewStub.setVisibility(View.GONE);
//        if (constraintLayout != null){
//            constraintLayout.setVisibility(View.GONE);
//        }
    }
    public void show(View view){
        viewStub.setVisibility(View.VISIBLE);
//        if (constraintLayout != null){
//            constraintLayout.setVisibility(View.VISIBLE);
//        }
    }
}

3.當呼叫第二次inflate的時候,會報錯:

編輯切換為居中

新增圖片註釋,不超過 140 字(可選)

我們看一下這是為什麼?進入viewStub.inflate();的原始碼:

public View inflate() {
        final ViewParent viewParent = getParent();
        if (viewParent != null && viewParent instanceof ViewGroup) {
            if (mLayoutResource != 0) {
                final ViewGroup parent = (ViewGroup) viewParent;
                final View view = inflateViewNoAdd(parent);
                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");
        }
    }

可以看到當viewParent為空或者不是viewgroup時才會報這個錯誤;那麼第一次呼叫的時候,肯定是進去了;發現一個方法replaceSelfWithView(view,parent);view就是我們在佈局檔案中給viewstub指定的layout所參照的那個佈局;parent就是getParent方法得到的,也就是acticity的填充佈局LinearLayout;

進去看一下:

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);
        }
    }

可以發現parent.removeViewInLayout(this);把this就是viewstub從父佈局linearlayout中移除了;parent.addView()就是把view(也就是我們參照的佈局)新增到了父佈局LinearLayout中。

我們用layout inspector來檢視一下:

inflate前:可以看到viewstub是灰色的

編輯

新增圖片註釋,不超過 140 字(可選)

inflate後:可以看到viewstub直接被移除了,把參照佈局直接放到view樹裡了。

編輯

新增圖片註釋,不超過 140 字(可選)

所以當我們第二次再呼叫inflate方法時,viewstub的parent已經為空了;就會丟擲此異常;

當呼叫textView = viewStub.findViewById(R.id.hello_tv);//獲取到的textview是空的;

而使用textView = findViewById(R.id.hello_tv);就可以直接拿到控制元件物件了;

當實現參照佈局的顯示和隱藏時,測試發現使用viewstub的setVisibility()方法可以實現,這是為什麼呢?;按理說使用constraintLayout.setVisibility()當然也可以;根據上面的view樹結構來看,好像使用參照佈局的setVisibility()方法更合理一些;

下面我們再來看看viewstub的setVisibility()為什麼也可以;跟進原始碼看看:

編輯切換為居中

新增圖片註釋,不超過 140 字(可選)

原始碼中使用mInflatedViewRef獲取到view,然後設定隱藏與顯示;mInflatedViewRef是一個view的弱參照WeakReference

其實在上面的inflate方法中已經為其新增了mInflatedViewRef = new WeakReference<>(view);這個view就是viewstub中的參照佈局;

所以,使用viewstub可以實現相同的顯示或隱藏效果;

從上圖的最後一個紅色框中可以發現,假設現在我沒有呼叫inflate方法,而是直接點選了show按鈕;然後參照佈局也可以繪製出來;這就是我在寫demo的時候,直接上去點選show按鈕,竟然也可以顯示的原因。

編輯切換為居中

新增圖片註釋,不超過 140 字(可選)

以上就是Android ViewStub的使用與簡單的演練;如果想要進階自己Android技能,可以參考這份《Android核心技術筆記》裡面記錄有Android的核心技術與其他前沿技術。

文末

Android ViewStub的使用注意事項

inflate方法只能呼叫一次,再次呼叫會出異常 我們看下inflate方法的原始碼,一旦第二次呼叫inflate方法,我們的到viewParent將等於null,會報 throw new IllegalStateException(“ViewStub must have a non-null ViewGroup viewParent”);異常,所以總之一句話,這個懶載入只能載入一次

public View inflate() {
       final ViewParent viewParent = getParent();
       if (viewParent != null && viewParent instanceof ViewGroup) {
           if (mLayoutResource != 0) {
               final ViewGroup parent = (ViewGroup) viewParent;
               final View view = inflateViewNoAdd(parent);
               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");
       }
   }

以上就是Android ViewStub使用方法學習的詳細內容,更多關於Android ViewStub使用的資料請關注it145.com其它相關文章!


IT145.com E-mail:sddin#qq.com