Android Saga| 探索View Binding

2020-03-26  本文已影响0人  Alwways

原文:Exploring View Binding on Android

在安卓应用程序中操作UI时,我们有2种选择。操作视图 (View)首先要获取视图的引用,对应有2种方法。一是findViewById()查询资源ID关联的View,然后将View强制转换成关联类型,Kotlin版本findViewById<>()运用reified和inline保证了类型安全,二是"kotlin-android-extension"编译时生成访问代码,以便根据id直接访问layout XML的View,合成代码的View Cache会将findViewById强转的结果缓存到HashMap。这两种方法都有用,不过都有些小坑。

View Binding将layout定义的views绑定到一个生成类给activity/fragment用,然后就能从绑定里访问可到达的视图。它直接带来下面这些好处。

视图绑定不支持布局变量或布局表达式,因此它不能用于在 XML 中将布局与数据绑定。

有了上面这些认知,就能理解为什么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一下就好了。

上一篇下一篇

猜你喜欢

热点阅读