前段时间遇见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。