顾名思义ViewBinding
的意思就是如何将view与代码绑定在一块儿。因此其主要解决如何安全优雅地从代码中引用到XML layout
文件中的view控件的问题。直到目前为止,Android构建用户界面的主流方式仍然是使用XML格式的layout
文件。java
视图访问的方式有经常使用的findViewById,ButterKnife等多种方式,这些方式的各方面对好比下android
谷歌在Android Studio 3.6 Canary 11
版本中正式推出视图绑定View Binding
安全
首先须要使用AS 3.6 Canary 11之上的版本,这里我使用AS 3.6.1
升级gradle plugin版本到3.6.1
markdown
ViewBinding
是一项功能,使您能够更轻松地编写与视图交互的代码。在模块中启用视图绑定后,它将为该模块中存在的每一个XML
布局文件生成一个绑定类。绑定类的实例包含对在相应布局中具备ID
的全部视图的直接引用。在大多数状况下,视图绑定替换findViewById()
。app
viewBinding
元素添加到其 build.gradle
文件中,以下例所示:android { ... viewBinding{ enabled = true } } 复制代码
tools:viewBindingIgnore="true"
属性添加到相应布局文件的根视图中:<LinearLayout
...
tools:viewBindingIgnore="true" >
...
</LinearLayout>
复制代码
为某个模块启用视图绑定功能后,系统会为该模块中包含的每一个 XML 布局文件各生成一个绑定类。每一个绑定类均包含对根视图以及具备 ID 的全部视图的引用。系统会经过如下方式生成绑定类的名称:将 XML 文件的名称转换为驼峰式大小写,并在末尾添加“Binding”一词。ide
activity_main.xml
:<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context=".MainActivity">
<TextView
android:id="@+id/tv_text"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:layout_marginTop="150dp"/>
<Button
android:id="@+id/btn"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="30dp"
/>
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"/>
</LinearLayout>
复制代码
生成的绑定类将名为
ActivityMainBinding
。此类具备两个字段:一个是名为tvText
的TextView
,另外一个是名为btn
的Button
。该布局中的ImageView
没有 ID,所以绑定类中不存在对它的引用。布局
每一个绑定类还包含一个
getRoot()
方法,用于为相应布局文件的根视图提供直接引用。在此示例中,ActivityMainBinding
类中的getRoot()
方法会返回LinearLayout
根视图。gradle
inflate()
方法。一般状况下,还能够调用 setContentView()
,从而将该绑定类的根视图做为参数进行传递,以使它成为屏幕上的活动视图。在此示例中,您能够在 Activity 中调用 ActivityMainBinding.inflate()
:public class MainActivity extends AppCompatActivity { private ActivityMainBinding mBinding; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); mBinding = ActivityMainBinding.inflate(getLayoutInflater()); setContentView(mBinding.getRoot()); mBinding.tvText.setText("是否是这样使用呢?"); mBinding.btn.setText("我是一个按钮"); } } 复制代码
findViewById
相比,视图绑定具备一些很显著的优势:@Nullable
标记。ViewBinding
时生成的ActivityMainBinding类
。ActivityMainBinding类
代码以下:原理就是Google在那个用来编译的gradle插件中增长了新功能,当某个module开启
ViewBinding
功能后,编译的时候就去扫描此模块下的layout
文件,生成对应的binding类。那些你所熟悉的findViewById
操做都是在这个自动生成的类里面呢,以下所示ui
// Generated by view binder compiler. Do not edit! package com.example.viewbinding.databinding; public final class ActivityMainBinding implements ViewBinding { @NonNull private final LinearLayout rootView; @NonNull public final Button btn; @NonNull public final TextView tvText; private ActivityMainBinding(@NonNull LinearLayout rootView, @NonNull Button btn, @NonNull TextView tvText) { this.rootView = rootView; this.btn = btn; this.tvText = tvText; } @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); } @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: { Button btn = rootView.findViewById(R.id.btn); if (btn == null) { missingId = "btn"; break missingId; } TextView tvText = rootView.findViewById(R.id.tv_text); if (tvText == null) { missingId = "tvText"; break missingId; } return new ActivityMainBinding((LinearLayout) rootView, btn, tvText); } throw new NullPointerException("Missing required view with ID: ".concat(missingId)); } } 复制代码
其中核心代码是
bind(@NonNull View rootView)
方法,除此以外还有两个inflate()重载方法,通常状况下咱们使用这两个方法得到binding类的实例,这些方法都是public static的,经过bind(@NonNull View rootView)这个方法应该能够实现延迟绑定,可是其使用场景应该不多。this