View Binding是Android Studio 推出的特性,目的是为了替代findViewById(内部实现还是使用findViewById)。在模块中启用视图绑定之后,系统会为该模块中的每个 XML 布局文件生成一个绑定类。绑定类的实例包含对在相应布局中具有 ID 的所有视图的直接引用。
注意:视图绑定在 Android Studio 3.6 Canary 11 及更高版本中可用。
View Binding 的优点
与使用 findViewById 相比,视图绑定具有一些很显著的优点:
- Null 安全:由于视图绑定会创建对视图的直接引用,因此不存在因视图 ID 无效而引发 Null 指针异常的风险。此外,如果视图仅出现在布局的某些配置中,则绑定类中包含其引用的字段会使用 @Nullable 标记。
- 类型安全:每个绑定类中的字段均具有与它们在 XML 文件中引用的视图相匹配的类型,这意味着不存在发生类转换异常的风险。例如 findViewById, ButterKnife 方式不小心将一个TextView错误的赋值给一个Button变量,都会报错,关键在错误还出现在运行时,而不是编译时。
- 更快的编译速度:视图绑定不需要处理注释,因此编译时间更短。
- 易于使用:视图绑定不需要特别标记的 XML 布局文件,因此在应用中采用速度更快。在模块中启用视图绑定后,它会自动应用于该模块的所有布局。
这些差异意味着布局和代码之间的不兼容将会导致构建在编译时(而非运行时)失败。
配置
首先ViewBinding默认是没有开启的,要在build.grade中添加如下代码开启ViewBinding机制
android {
...
// Android Studio 4.0
buildFeatures {
viewBinding = true
}
// Android Studio 3.6
viewBinding {
enabled = true
}
....
}
开启后,在编译的时候编译器就会为此模块下的每个layout文件都产生一个对应的绑定类,绑定类路径为:build\generated\data_binding_base_class_source_out\debug\out\包名\databinding\XXXXBinding.java
可以清楚的看到还是使用findViewById的方法实现绑定,注意:每个绑定类都包含对根视图和所有具有 ID 的视图的引用,如果视图中没有指定id,此绑定类就不会包含此元素的引用
如果希望在生成绑定类时忽略某个布局文件,将 tools:viewBindingIgnore=“true” 属性添加到相应布局文件的根视图中
<LinearLayout
...
tools:viewBindingIgnore="true" >
...
</LinearLayout>
使用
Activity
import com.zwt.zwtbindview.databinding.ActivityMainBinding;
public class MainActivity extends AppCompatActivity {
private ActivityMainBinding binding;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// setContentView(R.layout.activity_main);
binding = ActivityMainBinding.inflate(getLayoutInflater());
View view = binding.getRoot();
setContentView(view);
binding.helloButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
binding.helloText.setText("I am fine");
}
});
}
- inflate() :此操作会创建该绑定类的实例以供 Activity 使用
- getRoot():为相应布局文件的根视图提供直接引用(获取对根视图的引用)
- 将根视图传递到 setContentView(),使其成为屏幕上的活动视图。
Fragment
public class MyFragment extends Fragment {
private FragmentMyBinding binding;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
// return inflater.inflate(R.layout.fragment_my, container, false);
binding = FragmentMyBinding.inflate(inflater, container, false);
return binding.getRoot();
}
@Override
public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
super.onViewCreated(view, savedInstanceState);
binding.myText.setText("Hello !!!"); //布局文件中有一个 TextView 组件,id为myText
}
}
- 调用 inflate() 包含在生成的绑定类中的静态方法。这会为要使用的Fragment创建绑定类的实例。
- 通过调用getRoot()方法获取对根视图的引用。
- 从onCreateView()方法返回根视图,使其成为屏幕上的活动视图。
封装基类
利用反射
public class BaseActivity<T extends ViewBinding> extends AppCompatActivity {
protected T viewBinding;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ParameterizedType type = (ParameterizedType) getClass().getGenericSuperclass();
assert type != null;
Class cls = (Class) type.getActualTypeArguments()[0];
try {
Method infalte = cls.getDeclaredMethod("inflate", LayoutInflater.class);
viewBinding = (T) infalte.invoke(null, getLayoutInflater());
assert viewBinding != null;
setContentView(viewBinding.getRoot());
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
}
}
/*********************调用**************************/
public class MainActivity extends BaseActivity<ActivityMainBinding> {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// setContentView(R.layout.activity_main);
viewBinding.helloButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
//helloText.setText("I am fine");
viewBinding.helloText.setText("I am fine");
}
});
}
}
Fragment 与 Activity 类似
public class BaseFragment<T extends ViewBinding> extends Fragment {
protected T viewBinding;
@Nullable
@Override
public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
ParameterizedType type = (ParameterizedType) getClass().getGenericSuperclass();
assert type != null;
Class cls = (Class) type.getActualTypeArguments()[0];
try {
Method inflate = cls.getDeclaredMethod("inflate",LayoutInflater.class, ViewGroup.class, boolean.class);
viewBinding = (T) inflate.invoke(null, inflate, container, false);
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
assert viewBinding != null;
return viewBinding.getRoot();
}
}
/*********************调用**************************/
public class MyFragment extends BaseFragment<FragmentMyBinding > {
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
//不需要写onCreateView方法绑定视图
@Override
public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
super.onViewCreated(view, savedInstanceState);
viewBinding.myText.setText("Hello !!!"); //布局文件中有一个 TextView 组件,id为myText
}
}
非反射
public abstract class BaseActivity<T extends ViewBinding> extends AppCompatActivity {
protected T viewBinding;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
viewBinding = getViewBinding();
setContentView(viewBinding.getRoot());
}
protected abstract T getViewBinding();
}
/*********************调用**************************/
public class MainActivity extends BaseActivity<ActivityMainBinding> {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// setContentView(R.layout.activity_main);
viewBinding.helloButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
helloText.setText("I am fine");
viewBinding.helloText.setText("I am fine");
}
});
}
@Override
protected ActivityMainBinding getViewBinding() {
return ActivityMainBinding.inflate(getLayoutInflater());
}
}
public abstract class BaseFragment<T extends ViewBinding> extends Fragment {
protected T viewBinding;
@Nullable
@Override
public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
viewBinding = getViewBinding(inflater, container);
assert viewBinding != null;
return viewBinding.getRoot();
}
protected abstract T getViewBinding(LayoutInflater inflater, ViewGroup container);
}
/*********************调用**************************/
public class MyFragment extends BaseFragment<FragmentMyBinding> {
private FragmentMyBinding binding;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
@Override
protected FragmentMyBinding getViewBinding(LayoutInflater inflater, ViewGroup container) {
return FragmentMyBinding.inflate(inflater, container, false);
}
@Override
public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
super.onViewCreated(view, savedInstanceState);
binding.myText.setText("Hello !!!");
}
}