我已经阅读了Romain Guy在<merge />
标记上的帖子,但是我仍然不明白它的用处。它是<Frame />
标签的某种替换,还是这样使用:
<merge xmlns:android="....">
<LinearLayout ...>
.
.
.
</LinearLayout>
</merge>
那么<include />
代码在另一个文件中?
我已经阅读了Romain Guy在<merge />
标记上的帖子,但是我仍然不明白它的用处。它是<Frame />
标签的某种替换,还是这样使用:
<merge xmlns:android="....">
<LinearLayout ...>
.
.
.
</LinearLayout>
</merge>
那么<include />
代码在另一个文件中?
Answers:
<merge/>
之所以有用,是因为它可以摆脱不必要的ViewGroup,即仅用于包装其他视图且本身无用的布局。
例如,如果<include/>
要从另一个文件中进行布局而不使用合并,则这两个文件可能看起来像这样:
layout1.xml:
<FrameLayout>
<include layout="@layout/layout2"/>
</FrameLayout>
layout2.xml:
<FrameLayout>
<TextView />
<TextView />
</FrameLayout>
在功能上等效于以下单个布局:
<FrameLayout>
<FrameLayout>
<TextView />
<TextView />
</FrameLayout>
</FrameLayout>
layout2.xml中的那个FrameLayout可能没有用。 <merge/>
帮助摆脱它。这是使用合并的样子(layout1.xml不变):
layout2.xml:
<merge>
<TextView />
<TextView />
</merge>
这在功能上等效于以下布局:
<FrameLayout>
<TextView />
<TextView />
</FrameLayout>
但是由于您正在使用<include/>
,因此可以在其他地方重复使用布局。它不必用于仅替换FrameLayouts -您可以使用它来替换不会为视图的外观/行为添加任何有用内容的任何布局。
该<include>
标签可让您将布局划分为多个文件:它可以帮助处理复杂的或超长用户界面。
假设您使用两个包含文件来分割复杂的布局,如下所示:
top_level_activity.xml:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/layout1"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >
<!-- First include file -->
<include layout="@layout/include1.xml" />
<!-- Second include file -->
<include layout="@layout/include2.xml" />
</LinearLayout>
然后,您需要编写include1.xml
和include2.xml
。
请记住,包含文件中的xml只是在呈现时转储到top_level_activity
布局中(非常类似于#INCLUDE
C 的宏)。
包含文件是纯简布局xml。
include1.xml:
<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/textView1"
android:text="First include"
android:textAppearance="?android:attr/textAppearanceMedium"/>
...和include2.xml:
<?xml version="1.0" encoding="utf-8"?>
<Button xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/button1"
android:text="Button" />
看到?没有什么花哨。请注意,您仍然必须使用声明android名称空间xmlns:android="http://schemas.android.com/apk/res/android
。
因此,top_level_activity.xml的呈现版本 为:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/layout1"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >
<!-- First include file -->
<TextView
android:id="@+id/textView1"
android:text="First include"
android:textAppearance="?android:attr/textAppearanceMedium"/>
<!-- Second include file -->
<Button
android:id="@+id/button1"
android:text="Button" />
</LinearLayout>
在您的Java代码中,所有这些都是透明的:findViewById(R.id.textView1)
在活动类中,返回正确的小部件(即使该小部件是在与活动布局不同的xml文件中声明的)。
最重要的是:可视化编辑器可以轻松处理事情。顶层布局呈现与包含的XML。
由于包含文件是经典布局xml文件,因此它必须具有一个top元素。因此,如果您的文件需要包含多个控件,则必须使用一种布局。
假设include1.xml
现在有两个TextView
:必须声明一个布局。让我们选择一个LinearLayout
。
include1.xml:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/layout2"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >
<TextView
android:id="@+id/textView1"
android:text="Second include"
android:textAppearance="?android:attr/textAppearanceMedium"/>
<TextView
android:id="@+id/textView2"
android:text="More text"
android:textAppearance="?android:attr/textAppearanceMedium"/>
</LinearLayout>
该top_level_activity.xml将呈现为:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/layout1"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >
<!-- First include file -->
<LinearLayout
android:id="@+id/layout2"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >
<TextView
android:id="@+id/textView1"
android:text="Second include"
android:textAppearance="?android:attr/textAppearanceMedium"/>
<TextView
android:id="@+id/textView2"
android:text="More text"
android:textAppearance="?android:attr/textAppearanceMedium"/>
</LinearLayout>
<!-- Second include file -->
<Button
android:id="@+id/button1"
android:text="Button" />
</LinearLayout>
但是等两个级别LinearLayout
都是多余的!
确实,嵌套的两个LinearLayout
没有任何作用,因为TextView
可以layout1
为完全相同的呈现将两个嵌套在下面。
所以,我们能做些什么?
该<merge>
标签只是一个虚拟标签,它提供一个顶级元素来处理这种冗余问题。
现在include1.xml变为:
<merge xmlns:android="http://schemas.android.com/apk/res/android">
<TextView
android:id="@+id/textView1"
android:text="Second include"
android:textAppearance="?android:attr/textAppearanceMedium"/>
<TextView
android:id="@+id/textView2"
android:text="More text"
android:textAppearance="?android:attr/textAppearanceMedium"/>
</merge>
现在top_level_activity.xml呈现为:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/layout1"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >
<!-- First include file -->
<TextView
android:id="@+id/textView1"
android:text="Second include"
android:textAppearance="?android:attr/textAppearanceMedium"/>
<TextView
android:id="@+id/textView2"
android:text="More text"
android:textAppearance="?android:attr/textAppearanceMedium"/>
<!-- Second include file -->
<Button
android:id="@+id/button1"
android:text="Button" />
</LinearLayout>
您保存了一个层次结构级别,避免了一个无用的观点:Romain Guy已经睡得更好了。
你现在不快乐吗?
RelativeLayout
手动使用或绘制视图
为了更深入地了解正在发生的事情,我创建了以下示例。看一下activity_main.xml和content_profile.xml文件。
activity_main.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<include layout="@layout/content_profile" />
</LinearLayout>
content_profile.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Howdy" />
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Hi there" />
</LinearLayout>
在这里,膨胀后的整个布局文件如下所示。
<LinearLayout>
<LinearLayout>
<TextView />
<TextView />
</LinearLayout>
</LinearLayout>
看到父级LinearLayout内部有一个LinearLayout,它没有任何作用,并且是多余的。通过“布局检查器”工具查看布局可以清楚地解释这一点。
更新代码以使用合并而不是使用诸如LinearLayout之类的ViewGroup之后的content_profile.xml。
<merge xmlns:android="http://schemas.android.com/apk/res/android">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Howdy" />
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Hi there" />
</merge>
现在我们的布局看起来像这样
<LinearLayout>
<TextView />
<TextView />
</LinearLayout>
在这里,我们看到多余的LinearLayout ViewGroup被删除。现在,“布局检查器”工具提供了以下布局层次结构。
因此,当父布局可以放置子布局时,请始终尝试使用合并,或者当您了解层次结构中将有一个冗余视图组时,请更准确地使用合并。
使用合并的另一个原因是在ListViews或GridViews中使用自定义视图组时。您可以使用自定义视图,而不是在列表适配器中使用viewHolder模式。定制视图将为根为合并标记的xml膨胀。适配器代码:
public class GridViewAdapter extends BaseAdapter {
// ... typical Adapter class methods
@Override
public View getView(int position, View convertView, ViewGroup parent) {
WallpaperView wallpaperView;
if (convertView == null)
wallpaperView = new WallpaperView(activity);
else
wallpaperView = (WallpaperView) convertView;
wallpaperView.loadWallpaper(wallpapers.get(position), imageWidth);
return wallpaperView;
}
}
这是自定义视图组:
public class WallpaperView extends RelativeLayout {
public WallpaperView(Context context) {
super(context);
init(context);
}
// ... typical constructors
private void init(Context context) {
View.inflate(context, R.layout.wallpaper_item, this);
imageLoader = AppController.getInstance().getImageLoader();
imagePlaceHolder = (ImageView) findViewById(R.id.imgLoader2);
thumbnail = (NetworkImageView) findViewById(R.id.thumbnail2);
thumbnail.setScaleType(ImageView.ScaleType.CENTER_CROP);
}
public void loadWallpaper(Wallpaper wallpaper, int imageWidth) {
// ...some logic that sets the views
}
}
这是XML:
<merge xmlns:android="http://schemas.android.com/apk/res/android">
<ImageView
android:id="@+id/imgLoader"
android:layout_width="30dp"
android:layout_height="30dp"
android:layout_centerInParent="true"
android:src="@drawable/ico_loader" />
<com.android.volley.toolbox.NetworkImageView
android:id="@+id/thumbnail"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
</merge>
<TextView />
,而不包含其他任何内容。