背景
开发项目的时候,需要使用最新的依赖库版本,在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属性了。
原理
我们可以查看编译后生成的文件
打开该文件会看到,定义了三个常量,常量的类型刚好是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布局中引用的视图类型相匹配。这意味不存在类型转换的异常