<include> <merge> <view stub>使用以及为什么使用

本文详细介绍了Android中`include`、`merge`和`ViewStub`三种布局优化技术的使用方法及其优化点。`include`用于复用布局,减少代码冗余;`merge`避免了多余的父布局,提高性能;`ViewStub`则实现了布局的延迟加载,提升应用启动速度。通过合理运用这些技术,可以有效优化Android应用的布局结构和性能。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

布局使用的时候 可以多考虑下 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了。

解析日志 07-01 11:39:12.525 2846 2866 V WindowManager: Sent Transition (#1093) createdAt=07-01 11:39:11.551 via request=TransitionRequestInfo { type = OPEN, triggerTask = TaskInfo{userId=0 taskId=851 displayId=0 isRunning=true baseIntent=Intent { act=android.intent.action.MAIN cat=[android.intent.category.LAUNCHER] flg=0x10200000 pkg=com.google.android.apps.maps cmp=com.google.android.apps.maps/com.google.android.maps.MapsActivity } baseActivity=ComponentInfo{com.google.android.apps.maps/com.google.android.maps.MapsActivity} topActivity=ComponentInfo{com.google.android.apps.maps/com.google.android.maps.MapsActivity} origActivity=null realActivity=ComponentInfo{com.google.android.apps.maps/com.google.android.maps.MapsActivity} numActivities=1 lastActiveTime=283501092 supportsMultiWindow=true resizeMode=2 isResizeable=true minWidth=-1 minHeight=-1 defaultMinSize=220 token=WCT{RemoteToken{d374785 Task{d2792a5 #851 type=standard A=10163:com.google.android.apps.maps}}} topActivityType=1 pictureInPictureParams=PictureInPictureParams( aspectRatio=3/4 expandedAspectRatio=null sourceRectHint=null hasSetActions=false hasSetCloseAction=false isAutoPipEnabled=false isSeamlessResizeEnabled=false title=null subtitle=null isLaunchIntoPip=false) shouldDockBigOverlays=false launchIntoPipHostTaskId=-1 lastParentTaskIdBeforePip=-1 displayCutoutSafeInsets=Rect(0, 100 - 0, 0) topActivityInfo=ActivityInfo{e3229da com.google.android.maps.MapsActivity} launchCookies=[] positionInParent=Point(0, 0) parentTaskId=-1 isFocused=false isVisible=false isVisibleRequested=false isSleeping=false locusId=null displayAreaFeatureId=1 isTopActivityTransparent=false appCompatTaskInfo=AppCompatTaskInfo { topActivityInSizeCompat=false topActivityEligibleForLetterboxEducation= falseisLetterboxEducationEnabled= false isLetterboxDoubleTapEnabled= false topActivityEligibleForUserAspectRatioButton= false topActivityBoundsLetterboxed= false isFromLetterboxDoubleTap= false topActivityLetterboxVerticalPosition= -1 topActivityLetterboxHorizontalPosition= -1 topActivityLetterboxWidth=1080 topActivityLetterboxHeight=2400 isUserFullscreenOverrideEnabled=false isSystemFullscreenOverrideEnabled=false cameraCompatTaskInfo=CameraCompatTaskInfo { cameraCompatControlState=hidden freeformCameraCompatMode=inactive}}}, pipTask = null, remoteTransition = RemoteTransition { remoteTransition = android.window.IRemoteTransition$Stub$Proxy@a5594d1, appThread = null, debugName = SysUILaunch }, displayChange = null, flags = 0, debugId = 1093 }
最新发布
07-17
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值