美文网首页
Android Saga| 探索View Binding

Android Saga| 探索View Binding

作者: Alwways | 来源:发表于2020-03-26 15:41 被阅读0次

    原文: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。这两种方法都有用,不过都有些小坑。

    • 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一下就好了。

    相关文章

      网友评论

          本文标题:Android Saga| 探索View Binding

          本文链接:https://www.haomeiwen.com/subject/icecuhtx.html