Android Jetpack ViewBinding

作者: 一个有故事的程序员 | 来源:发表于2021-11-04 16:26 被阅读0次

    导语

    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));
    }
    

    更多内容戳这里(整理好的各种文集)

    相关文章

      网友评论

        本文标题:Android Jetpack ViewBinding

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