导语
Jetpack简介及其它组件文章
还在因为findViewById而烦恼吗?那就使用ViewBinding吧。
主要内容
- 什么是ViewBinding
- ViewBinding的优势
- 其它方案现状
- ViewBinding的使用
具体内容
什么是ViewBinding
通过视图绑定功能,您可以更轻松地编写可与视图交互的代码。在模块中启用视图绑定之后,系统会为该模块中的每个 XML 布局文件生成一个绑定类。绑定类的实例包含对在相应布局中具有 ID 的所有视图的直接引用。在大多数情况下,视图绑定会替代 findViewById。
简单来讲,就是可以简单的直接调用View,不需要findViewById了。
ViewBinding的优势
- 不再使用findViewById,提升开发效率,减少代码冗余。
- 不用担心View为null,也不需要强转类型,更加安全。
其它方案现状
- Butter Knife的作者已经不再维护,Butter Knife使用了反射的方式。
- Kotlin Android Extensions也已经被Google废弃,即便使用了HashMap这种时间复杂度为O(1)的方式存储查询,但还是会降低一定的效率。
- DataBinding相比ViewBinding编译较慢一些,但是可以使用双向数据绑定,如果不使用DataBinding的其它功能的话,建议直接使用ViewBinding。
ViewBinding的使用
Android Studio 3.6 Canary 11 及更高版本中可用:
android {
viewBinding {
enabled = true
}
}
从Android Gradle Plugin 4.0.0-alpha05这里开始,有一个称为buildFeatures启用构建功能的新块。
android {
buildFeatures {
// 启用 viewBinding
viewBinding = true
}
}
为某个模块启用视图绑定功能后,系统会为该模块中包含的每个 XML 布局文件生成一个绑定类。每个绑定类均包含对根视图以及具有 ID 的所有视图的引用。系统会通过以下方式生成绑定类的名称:将 XML 文件的名称转换为驼峰式大小写,并在末尾添加“Binding”一词。
例如,假设某个布局文件的名称为 result_profile.xml:
<LinearLayout ... >
<TextView android:id="@+id/name" />
<ImageView android:cropToPadding="true" />
<Button android:id="@+id/button"
android:background="@drawable/rounded_button" />
</LinearLayout>
所生成的绑定类的名称就为 ResultProfileBinding。此类具有两个字段:一个是名为 name 的 TextView,另一个是名为 button 的 Button。该布局中的 ImageView 没有 ID,因此绑定类中不存在对它的引用。
每个绑定类还包含一个 getRoot() 方法,用于为相应布局文件的根视图提供直接引用。在此示例中,ResultProfileBinding 类中的 getRoot() 方法会返回 LinearLayout 根视图。
以下几个部分介绍了生成的绑定类在 Activity 和 Fragment 中的使用。
在 Activity 中使用视图绑定
private ResultProfileBinding binding;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
binding = ResultProfileBinding.inflate(getLayoutInflater());
View view = binding.getRoot();
setContentView(view);
}
private void setDataToView() {
binding.getName().setText(viewModel.getName());
binding.button.setOnClickListener(new View.OnClickListener() {
viewModel.userClicked()
});
}
在 Fragment 中使用视图绑定
private ResultProfileBinding binding;
@Override
public View onCreateView (LayoutInflater inflater,
ViewGroup container,
Bundle savedInstanceState) {
binding = ResultProfileBinding.inflate(inflater, container, false);
View view = binding.getRoot();
return view;
}
@Override
public void onDestroyView() {
super.onDestroyView();
binding = null;
}
private void setDataToView() {
binding.getName().setText(viewModel.getName());
binding.button.setOnClickListener(new View.OnClickListener() {
viewModel.userClicked()
});
}
注意:Fragment 的存在时间比其视图长。请务必在 Fragment 的 onDestroyView() 方法中清除对绑定类实例的所有引用。
在 RecyclerView.Adapter 中使用视图绑定
public static class ViewHolder extends RecyclerView.ViewHolder{
public ItemBinding binding;
public ViewHolder(ItemBinding itemBinding) {
super(itemBinding.getRoot());
this.binding = itemBinding;
}
}
@NonNull
@Override
public ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
ItemBinding binding = ItemBinding.inflate(LayoutInflater.from(parent.getContext()), parent, false);
return new ViewHolder(binding);
}
@Override
public void onBindViewHolder(@NonNull ViewHolder holder, int position) {
holder.binding.tv.setText(mDataList.get(position));
}
网友评论