Android的视图绑定
视图绑定(ViewBinding)在开发中起到的作用是代替findViewById。
初始设置
ViewBinding是按模块启动的,在使用之前需要在模块中的gradle中开启ViewBinding。
//kotlin
android {
...
buildFeatures {
viewBinding = true
}
}
//groovy
android {
...
buildFeatures {
viewBinding true
}
}
使用
当我们为某个模块引入ViewBinding并Sync之后,系统会自动为该模块下的每一个xml生成一个绑定类,每个绑定类都包含该xml中的所有组件id的引用,而这些类的名称都与xml有关。例如我们已有一个布局文件名为result_profile.xml,他的绑定类名称就是ResultProfileBinding。
以下展示了如何在Activity和Fragment中使用生成的绑定类。
Activity
在onCreate中:
- 调用绑定类的inflate方法,目的是创建该绑定类的实例;
- 调用getRoot方法或使用Kotlin属性语法,获取对根视图的使用;
- 将第二步获取的根视图传递给setContentView方法,使其成为屏幕上的活动View。
//kotlin
private lateinit var binding: ResultProfileBinding
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
binding = ResultProfileBinding.inflate(layoutInflater)//1+2
setContentView(view)//3
}
//java
private ResultProfileBinding binding;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
binding = ResultProfileBinding.inflate(getLayoutInflater());//1
View view = binding.getRoot();//2
setContentView(view);//3
}
此后可以使用该绑定类的实例来引用任何视图:
//kotlin
private ResultProfileBinding binding;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
binding = ResultProfileBinding.inflate(getLayoutInflater());
View view = binding.getRoot();
setContentView(view);
}
//java
binding.name.setText(viewModel.getName());
binding.button.setOnClickListener(new View.OnClickListener() {
viewModel.userClicked()
});
Fragment
Fragment和Activity中的用法差不多,需要在onCreateView方法中执行以下:
- 调用绑定类中的inflate方法;
- 调用getRoot方法或使用Kotlin属性语法获取根视图的引用;
- 从onCreateView中返回根视图,使其成为屏幕上的活动View。
//kotlin
//该变量只在onCreateView和onDestroyView中可见
private var _binding: ResultProfileBinding? = null
private val binding get() = _binding!!
override fun onCreateView(
inflater: LayoutInflater,
container: viewGroup?,
savedInstanceState: Bundle?
): View? {
_binding = ResultProfileBinding.inflate(inflater, container, false)
val view = binding.root
return view
}
override fun onDestroyView() {
super.onDetroy()
_binding = null
}
//java
private ResultProfileBinding binding;
@Override
public View onCreateView (LayoutInflater inflater,
ViewGroup container,
Bundle savedInstanceState) {
binding = ResultProfileBinding.inflate(inflater, container, false);
View view = binding.getRoot();
return view;
}
@Override
public void onDestroyView() {
super.onDestroyView();
binding = null;
}
由于kotlin中空类型系统的存在,在fragment中的ViewBinding写法与java相比会稍显复杂,需要在新建一个变量接收绑定的实例,并给出非空断言。
与findViewById的对比
两者都可用于直接引用视图的绑定类,ViewBinding的优势在于:
- 加快编译速度:ViewBinding不需要处理注解
- 易于使用:不需要标记xml文件,因为每个xml都会自动生成相应的绑定类
而与findViewById即数据绑定相比,ViewBinding具有以下限制:
- 不支持布局变量或布局表达式,因此不能直接从xml声明动态界面内容
- 不支持双向数据绑定