<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
當渲染一個活動時,這個活動的佈局可能會有很多visible為invisible和gone的情況,雖然這些控制元件雖然現在不顯示在螢幕上,但是系統在載入這個佈局檔案時還是會載入它的,這就影響了這個頁面的載入效率,因為這些不可見的控制元件提前載入它們並沒有什麼實際的意義,反而會減緩頁面的載入時間,所以為了解決這個問題可以使用ViewStub來懶載入暫時不顯示的佈局.
簡單來說, ViewStub可以做到按需載入一個佈局,我們可以控制它載入的時機,而不是在Activity的onCreate方法中去載入.即懶載入
<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();
也就是在我們的app啟動繪製頁面的時候,他不會繪製到view樹中;當在程式碼中執行inflate操作後,她才會被新增到試圖中。其實ViewStub就是一個寬高都為0的一個View,它預設是不可見的,只有通過呼叫setVisibility函數或者Inflate函數才 會將其要裝載的目標佈局給載入出來,從而達到延遲載入的效果,這個要被載入的佈局通過android:layout屬性來設定。最終目的是把app載入頁面的速度提高了,使使用者體驗更好。
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); // } } }
編輯切換為居中
新增圖片註釋,不超過 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的核心技術與其他前沿技術。
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其它相關文章!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45