瞅瞅AS3.6新增ViewBinding
目录
- 基本使用
- 查看ViewBinding实现方式
- include,merge,viewstub的配合使用
开启流程
- 首先更新AS到3.6版本
- 打开module的gradle文件
- 在android函数里面添加如下代码,这样就开启完成了
android {
//...
viewBinding.enabled = true
}
使用
- 默认情况下布局XML文件都会生成一个 XXXBinding 类
- 如有不需要生成Binding类的,可在跟布局添加属性 tools:viewBindingIgnore="true"如:
<FrameLayout
tools:viewBindingIgnore="true">
</FrameLayout>
- 接着就可以在Activity里面进行使用 XXXBinding 类,该类提供了三个初始化函数
fun inflate(inflater: LayoutInflater): XXXBinding
fun inflate(inflater: LayoutInflater, parent: ViewGroup, attachToParent: boolean): XXXBinding
fun bind(view: View): XXXBinding
- 初始化后,可通过返回的XXXBinding类的getRoot函数获取布局的跟View,再通过setContentView添加到Activity
- XXXBinding实例的成员变量是以布局文件里View的ID来命名
- 附加两个例子图
以上就是基础用法
看看ViewBinding为我们做了什么
- 打开生成的XXXBinding类,文件路径为: {module_path}/build/generated/data_binding_base_class_source_out/debug/out/cc/xiaobaicz/test/databinding/XXXBinding.java
- 看看实现,比较简单的实现, 主要逻辑 在于bind函数内部,可以看到绑定过程做了非空判断
public final class ActivityMainBinding implements ViewBinding {
@NonNull
private final LinearLayout rootView;
@NonNull
public final Button btnCancel;
@NonNull
public final Button btnSubmit;
@NonNull
public final TextView tvMsg;
private ActivityMainBinding(@NonNull LinearLayout rootView, @NonNull Button btnCancel,
@NonNull Button btnSubmit, @NonNull TextView tvMsg) {
this.rootView = rootView;
this.btnCancel = btnCancel;
this.btnSubmit = btnSubmit;
this.tvMsg = tvMsg;
}
@Override
@NonNull
public LinearLayout getRoot() {
return rootView;
}
@NonNull
public static ActivityMainBinding inflate(@NonNull LayoutInflater inflater) {
return inflate(inflater, null, false);
}
@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);
}
//这里是主要的View绑定实现
@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: {
//绑定View且做了非空判断,抛出空指针异常
Button btnCancel = rootView.findViewById(R.id.btn_cancel);
if (btnCancel == null) {
missingId = "btnCancel";
break missingId;
}
Button btnSubmit = rootView.findViewById(R.id.btn_submit);
if (btnSubmit == null) {
missingId = "btnSubmit";
break missingId;
}
TextView tvMsg = rootView.findViewById(R.id.tv_msg);
if (tvMsg == null) {
missingId = "tvMsg";
break missingId;
}
return new ActivityMainBinding((LinearLayout) rootView, btnCancel, btnSubmit, tvMsg);
}
throw new NullPointerException("Missing required view with ID: ".concat(missingId));
}
}
include,merge,viewstub的配合使用
- 单独使用 include 的情况,可以直接给 include节点 添加id,XXXBinding类会直接把需要导入的布局的XXXBinding类做为成员变量(变量名为include标签的id)供开发者使用; (PS:ViewStub用法和include差不多,区别是 include 的成员变量是 XXXBinding,ViewStub成员变量就是ViewStub自己,然后用法跟原本ViewStub一致) 如:
- 带 merge 的 include
这个情况有点特殊,带 merge 标签的include不能使用ID,否则会找不到View报空指针异常
这个情况,我们可以先初始化主布局,再初始带 merge 的布局,如下示例:
好了,这就是ViewBing的基本使用~~