布局使用的时候 可以多考虑下 include merge view stub
各自使用如下以及优化的点
include
如果有的布局重复了 可以把重复的 用include 包裹起来当成个模块去利用
先写模块 test.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="wrap_content"
android:layout_height="wrap_content">
<TextView
android:id="@+id/tv_test"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="test"
android:textColor="@android:color/black"
android:textSize="13sp" />
</LinearLayout>
后利用
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="vertical">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="test_include"
android:textColor="@android:color/black"
android:textSize="13sp" />
<include layout="@layout/test" />
</LinearLayout>
但是,使用 include,include里面的布局又是一个 ViewGroup 的话,就会造成层级过多,
所以 include 里面的布局可以 使用 merge
merge使用
merge并不是一个ViewGroup,也不是一个View,它相当于声明了一些视图,等待被添加
merge标签被添加到父容器后,merge下的所有视图将被添加到父容器下,即merge下所有视图都会遵循父容器的布局规则
在这里
merge.xml
写着背景颜色和 垂直摆放都不会生效
下面的两个子view textview 只会遵循 引用 merge.xml 的父容器
也就是这里的 LinearLayout 的 水平摆放
merge 的优化点就在于 比如说 你有个布局 里面根布局是个 FrameLayout。
然后你这个布局又会加入到其他布局里面去。而加入的布局父容器也是 FrameLayout
所以你完全可以利用merge ,在子模块布局里面去掉 FrameLayout 写成 merge 利用 merge 特性 少掉 一层 FrameLayout
merge使用注意点:
- merge必须放在布局文件的根节点上,对merge设置的所有属性都无效
- merge并不是一个ViewGroup,也不是一个View,它相当于声明了一些视图,等待被添加。
- 因为merge标签并不是View,所以在通过LayoutInflate.inflate方法渲染的时候,第二个参数必须指定一个父容器,且第三个参数必须为true,也就是必须为merge下的视图指定一个父亲节点。
- merge标签被添加到父容器后,merge下的所有视图将被添加到父容器下,即merge下所有视图都会遵循父容器的布局规则
view stub 使用
我们有时根据需求,先把布局画好,然后把 android:visibility 设置成 “invisible” 或者是 “gone” ,invisible 和 gone 虽然看不见,但他们还是有初始化,占用这内存和资源,前者还占用着位置。
我们可以使用 ViewStub 来包裹这些需要隐藏显示的 view,它是一个轻量级的view,不可见不占用资源,只有当设置 inflate 时才初始化显示。
也是在布局 布局里面 引入 一个子模块布局
当然你也可以直接在 view stub 里面写你的布局
view stub的作用在于 可以利用 这个 view stub 来控制 内部的子view什么时候加载
也就是 activity_main.xml 这个布局 在 LayoutInflater.inflate 在创建view的时候 不会创建 view stub标签下的view ,等到执行
ViewStub viewStub = findViewById(R.id.vs_test);
if (viewStub != null) {
//viewStub.setVisibility(View.VISIBLE);
View inflated = viewStub.inflate();
View view = inflated.findViewById(R.id.tv_test);
}
也就是利用 viewStub 这个对象调用他的
viewStub.inflate() 或 viewStub.setVisibility(View.VISIBLE) 后,ViewStub标签下的布局才会显示出来,viewStub.inflate()不执行,页面不会展示也不会绘制ViewStub标签下的布局。
这个跟view 设置gone 还是有区别的 ,设置gone那种是 隐藏但是还是view会创建
viewstub 这种是 view不会被创建 所以也不会显示 调用 viewStub.inflate() 才会被创建显示
这种延迟手动创建 就可以加快了布局的启动数据提升性能
注:ViewStub只能被Inflate 或者 setVisibility(View.VISIBLE) 一次,inflate之后ViewStub对象会被置空,就不能够再通过ViewStub来控制显隐他标签下的view了。