踩了merge标签的一个坑

前段时间遇见merge标签的一个坑,其实也不是坑,而是自己使用不当造成的,这里给大家分享一下。

merge标签的作用应该都知道,它可以减少UI层级,优化布局。merge一般情况下配合include使用,很多地方说merge标签只能用于FramLayout和LinearLayout,实际上RelativeLayout也是可以使用merge的,但一般都不这么用,因为RelativeLayout本身位置关系复杂一点,处理不当很容易适得其反。

我踩坑的地方代码如下,一个LinearLayout里又包了几个layout,也都是LinearLayout,这几个layout也分别都是独立的业务,为了不让代码看起来一大坨,所以把他们拆出来放在不同的布局里,大致如下。

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">
  
    <include
        android:id="@+id/recommend_layout"
        layout="@layout/recommend_layout"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />
   
    <include
        android:id="@+id/daily_layout"
        layout="@layout/daily_best_layout"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />
  
    <include
        layout="@layout/dynamic_layout"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

</LinearLayout>

如果你知道这个坑或者踩过,仔细看上面的代码,会发现我有个地方已经出错了,但当时并没有意识到。以上面的recommend_layout为例,再随便写个recommend_layout的代码。

 <merge
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
       >
        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="center"
            android:text="this is test" />
        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="center"
            android:text="this is test" />
    </merge>

代码大致就是这样的,当某个条件触发的时候,会隐藏掉recommend_layout,但测试的过程中发现当条件出发后就崩溃了,空指针,找不到recommend_layout。刚开始我把recommend_layout申明为LinearLayout,以为是这个原因,改成View后很自信的commit甚至都没有测试。还是崩溃了,刚开始怀疑是因为数据请求跑到了View初始化前面或者View被回收了,断点了下发现都不是。排除了很久,没找到原因,久到我都忘了之前用了merge。后来恍然大悟:merge是个tag,不是view,findviewById能找到吗?把merge改成LinearLayout之后,OK了。

我们在xml中鼠标选中merge,Android studio会提示:Root tag “view”。这个view是标了引号的,虽然比较隐晦,但似乎是在告诉我们,它不是个view,而是个tag,所以像上面的案例无论我把id放在include里,还是放在merge标签包裹的layout里,都无法通过findviewbyId找到,因为他们俩都是tag,没有view,但是如果我弃用merge,改成LinearLayout就找得到了。

总结一下,merge它不是个view,而是个tag。Layout在inflate的时候遇到这个tag的时候就会跳过,并自动合并两个view层级,以此来减少不必要的view层级,关于这一点,可以通过hierarchy viewer查看。还有一点,merge只能用在跟布局。想了解更多其他标签的用法,欢迎来有心课堂。That is all。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 太长不看版:在 Android UI 布局过程中,遵守一些惯用、有效的布局原则,可以制作出高效且复用性高的 UI。...
    Mupceet阅读 3,902评论 0 14
  • 在开发过程中,往往会听到 “性能优化” 这个概念,这个概念很大,比如网络性能优化、耗电量优化等等,对我们开发者而言...
    湫水长天阅读 1,182评论 0 4
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,909评论 25 708
  • 近年来,关于用人单位拖欠劳动者报酬的案件逐年增多,劳动者在进行法律诉讼时,往往因为无法举出有力的证据而感到困惑。现...
    538a902176e9阅读 448评论 0 0
  • 有人告诉我说时间总会抹平一切包括无奈和痛苦 可是为什么50天已经过去我却更能清晰感觉到失去的心碎 在家的每一天看书...
    EliseTan阅读 321评论 0 1