ViewBinding应用

2020-04-24  本文已影响0人  烧伤的火柴

背景

开发项目的时候,需要使用最新的依赖库版本,在github上找Butter Knife的时候,看到了如下一段表述:
This tool is now deprecated. Please switch to [view binding]
Butter Knife现在已经废弃了,请切换到view binding,那么这个view binding是什么呢?连Jack大神都主动推荐。今天我们就一起看看这个组件。此文有些是借鉴 [Android视图绑定ViewBinding的使用](https://cloud.tencent.com/developer/article/1602245),先啰嗦的可以直接看这边博客。

介绍

通过ViewBinding,可以更轻松地编写可与视图交互的代码。
注意:viewbinding在 需要在[Android Studio 3.6 Canary 11 及更高版本]中可用。

使用流程

1.在modeule的build.gradle中打开viewBinding功能

android {
        ...
        viewBinding {
            enabled = true
        }
    }

开启后,编译器会为该module中的每个XML布局文件生成一个ViewBinding的是实现类。每个实现类都包含对根视图和具有id的视图属性,
如果你希望在绑定类时候忽略某个文件,可以使用 tools:viewBindingIgnore="true" 属性添加到相应布局文件的根视图中:

<LinearLayout
            ...
            tools:viewBindingIgnore="true" >
        ...
    </LinearLayout>

2.布局文件activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
   ...
    tools:context=".MainActivity">

    <TextView
        android:id="@+id/nameTV"
        .../>

    <TextView
        android:id="@+id/passwordTV"
       ... />

</androidx.constraintlayout.widget.ConstraintLayout>

完成布局文件后点击Make Projcet,编译工程,

3.在Activity中使用ViewBinding

class MainActivity : AppCompatActivity() {
    private lateinit var binding:ActivityMainBinding
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)

        binding = LayoutInflater.from(this).run {
            ActivityMainBinding.inflate(this)
        }
        setContentView(binding.root)

        binding.nameTV.text = "jawe"
        binding.passwordTV.text="123456"
    }
}

这里第一步通过ActivityMainBinding的静态方法inflate创建ActivityMainBinding的实例对象,然后通过setContentView设置root,然后就可以通过引用布局文件中的nameTV属性了。

原理

我们可以查看编译后生成的文件


viewBinding文件路径.png

打开该文件会看到,定义了三个常量,常量的类型刚好是rootView 对应activity_main.xml的根视图LinearLayout和两个定义了id的TextView,而且私有的构造方法

public final class ActivityMainBinding implements ViewBinding {
  @NonNull
  private final ConstraintLayout rootView;

  @NonNull
  public final TextView nameTV;

  @NonNull
  public final TextView passwordTV;

  private ActivityMainBinding(@NonNull ConstraintLayout rootView, @NonNull TextView nameTV,
      @NonNull TextView passwordTV) {
    this.rootView = rootView;
    this.nameTV = nameTV;
    this.passwordTV = passwordTV;
  }

 ...
}

构造方法是私有的,那么我们不能通过new创建实例了,类中还有一些静态方法创建实例的。


  @NonNull
  public static ActivityMainBinding inflate(@NonNull LayoutInflater inflater,
      @Nullable ViewGroup parent, boolean attachToParent) {
    View root = inflater.inflate(R.layout.activity_main, parent, false);
    if (attachToParent) {
      parent.addView(root);
    }
    return bind(root);
  }

  @NonNull
  public static ActivityMainBinding bind(@NonNull View rootView) {
    // The body of this method is generated in a way you would not otherwise write.
    // This is done to optimize the compiled bytecode for size and performance.
    String missingId;
    missingId: {
      TextView nameTV = rootView.findViewById(R.id.nameTV);
      if (nameTV == null) {
        missingId = "nameTV";
        break missingId;
      }
      TextView passwordTV = rootView.findViewById(R.id.passwordTV);
      if (passwordTV == null) {
        missingId = "passwordTV";
        break missingId;
      }
      return new ActivityMainBinding((ConstraintLayout) rootView, nameTV, passwordTV);
    }
    throw new NullPointerException("Missing required view with ID: ".concat(missingId));
  }

主要的逻辑就是通过inflate得到根视图rootView,然后使用rootView的findViewById方法找到所有的视图控件。

优点

与传统的findViewById相比有如下优点:
1.null安全:由于View Binding会创建对视图的直接引用,因此不存在因视图id无效引发null指针。此外,如果视图仅出现在布局的某些配置中,则绑定类中包含其引用的字段会使用 @Nullable 标记。
2.类型安全:每个绑定类中的字段都和他们在xml布局中引用的视图类型相匹配。这意味不存在类型转换的异常

上一篇下一篇

猜你喜欢

热点阅读