Android Saga| 探索View Binding
在安卓应用程序中操作UI时,我们有2种选择。操作视图 (View)首先要获取视图的引用,对应有2种方法。一是findViewById()
查询资源ID关联的View,然后将View强制转换成关联类型,Kotlin版本findViewById<>()
运用reified和inline保证了类型安全,二是"kotlin-android-extension"编译时生成访问代码,以便根据id直接访问layout XML的View,合成代码的View Cache会将findViewById强转的结果缓存到HashMap。这两种方法都有用,不过都有些小坑。
-
findViewById()
强转视图类型不是type safe的做法。这就是说可能声明View是一种类型,从layou引用的view是另一种类型,强制转换错误的视图类型引用就会导致ClassCastException
。 -
无论是用findViewId还是kotlin synthetics引用View,它们都不保证null safety。这就是说如果layout暂时还没有某个View或者用了错误的ID,访问的时候就会由于引用的view不存在而抛出NPE。
-
有时一个Acticity/Fragment有多个layout,不同的xml可能包含不同的view set,如果发生layout configuration改变,切换时没有处理nullable(可空)视图,用户使用时就会产生NPE。
View Binding将layout定义的views绑定到一个生成类给activity/fragment用,然后就能从绑定里访问可到达的视图。它直接带来下面这些好处。
视图绑定不支持布局变量或布局表达式,因此它不能用于在 XML 中将布局与数据绑定。
- 绑定类中的bound views根据实际关联的类型生成,从绑定类拿到视图的引用都是类型安全的。这就杜绝了不正确的强制类型转换,不会发生类型转换异常。
- 由于绑定类又是根据链接的layout生成,绑定类中所有视图都保证null safe,可以在activity/fragment中访问。这就是说不可能会从其他不可用的布局或视图中引用错误ID,也就不会出现NPE。
- 如果视图并非在所有布局中都可用,View Binding也能将视图标记为@Nullable。这就要求访问视图时要注意到视图可能为空,预防出错。
有了上面这些认知,就能理解为什么View Binding能够降低视图访问出错的可能,降低应用崩溃。
View Binding实现要两个条件。XML布局不需要动,不一样的是要在activity声明绑定类,为视图组件创建绑定,描述关联。
🔯注意:视图绑定在 Android Studio 3.6 Canary 11 及更高版本中可用。
使用View Binding特性,首先要在相应模块的build.gradle增加下面的配置:
android {
…
viewBinding {
enabled = true
}
}
视图绑定特定于模块,哪个模块用就往那个模块加配置,全局配置要放在项目build.gradle文件中。
layout文件add_profile.xml
<LinearLayout ... >
<TextView android:id="@+id/text_title" />
<Button android:id="@+id/button_add_profile" />
</LinearLayout>
给布局文件声明绑定类,只要遵守View Binding API的命名约定,add_profile.xml的绑定类像这样:
private lateinit var binding: AddProfileBinding
API按下划线分割布局文件名,每个单次首字母大写/标题格式,最后加上Binding后缀表示这是个绑定类。声明完绑定类,就要给它指定一个引用。将Acticity持有的LayoutInflator引用传入Binding类的静态方法inlate(),将布局展开成绑定类,暴露所有的bound views(Activity#getInfalator()返回LayoutInflator引用)。
@Override
fun onCreate(savedInstanceState: Bundle) {
super.onCreate(savedInstanceState)
binding = AddProfileBinding.inflate(layoutInflater)
setContentView(binding.root)
}
注意setContentView(binding.root)
的变更,每个绑定都有一个root——布局文件中的root,root可以用来直接finalize掉当前的screen。
使用绑定代替
binding.textTitle.text = getString(R.string.some_string)
binding.buttonAddProfile.setOnClickListener {
// do something
}
访问布局文件定义的组件 (去掉下划线的小驼峰) 屏蔽了XML中所有视图类型的差异,完全没有强制类型转换!同时提高了代码的可读性。还记得可空视图吗,看下面的招数。
binding.buttonAddProfile?.setOnClickListener {
// do something
}
如果要在生成绑定类时忽略某个布局文件,可以配置布局标签的viewBindingIgnore属性。
<LinearLayout
...
tools:viewBindingIgnore="true" >
...
</LinearLayout>
补充:如何在Fragment中使用视图绑定
直接上代码方便我复制
private FragmentSettingBinding binding;
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
// Inflate the layout for this fragment
binding = FragmentSettingBinding.inflate(inflater, container, false);
View view = binding.getRoot();
return view;
}
必须重写onDestroyView()
方法将binding对象置null。
@Override
public void onDestroyView() {
super.onDestroyView();
binding = null;
}
如果绑定类飘红了不要慌,Gradle Sync一下就好了。