前段时间更新ButterKnife,被R2给恶心到了,所以准备学习一下ViewBinding将ButterKnife替换掉。
引用
Android-ViewBinding入门
ViewBinding探索
ViewBinding的作用
在大部情况下,使用ViewBinding替换findViewById。以前如果不使用第三方框架, 不管怎样封装,使用起来还是很不方便(比如找不到控件、控件ID冲突等问题)
ViewBinding的开启流程
- 首先更新AS到3.6版本
- 打开module的gradle文件
- 在android函数里面添加如下代码,这样就开启完成了
android {
...
viewBinding {
enabled = true
}
}
ViewBinding的使用
- 默认情况下布局XML文件都会生成一个 XXXBinding 类
- 如有不需要生成Binding类的,可在跟布局添加属性 tools:viewBindingIgnore="true"如:
<LinearLayout
...
tools:viewBindingIgnore="true" >
...
</LinearLayout>
- 添加一个布局文件activity_main.xml,代码如下:
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout 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"
tools:context=".MainActivity">
<TextView
android:id="@+id/tv_content"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Hello World!"
android:textSize="24sp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent" />
</androidx.constraintlayout.widget.ConstraintLayout>
- 接着就可以在Activity里面进行使用 XXXBinding 类,该类提供了三个初始化函数
fun inflate(inflater: LayoutInflater): XXXBinding
fun inflate(inflater: LayoutInflater, parent: ViewGroup, attachToParent: boolean): XXXBinding
fun bind(view: View): XXXBinding
- 添加一个ActivityMainActivity.java,代码如下
public class MainActivity extends AppCompatActivity {
private ActivityMainBinding viewBinding;//ActivityMainBinding就是上面activity_main.xml生成的绑定文件
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//以前是这样绑定布局文件的
// setContentView(R.layout.activity_main);
//现在是这样绑定布局文件的
viewBinding = ActivityMainBinding.inflate(getLayoutInflater());
setContentView(viewBinding.getRoot());
//在这里设置一个显示文件
viewBinding.tvContent.setText("这里是通过ViewBinding设置的文本");
}
}
运行效果:
在Fragment在使用ViewBinding
- 先添加一个布局文件fragment_view_binding.xml,代码如下:
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent">
<TextView
android:id="@+id/tv_content"
android:layout_width="match_parent"
android:layout_height="0dp"
android:gravity="center"
android:text="Hello World!"
android:textSize="24sp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintVertical_weight="1" />
</androidx.constraintlayout.widget.ConstraintLayout>
- 添加一个Fragment,ViewBindingFragment,代码如下:
public class ViewBindingFragment extends Fragment {
private FragmentViewBindingBinding binding;
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
binding = FragmentViewBindingBinding.inflate(inflater, container, false);
View view = binding.getRoot();
binding.tvContent.setText("这里是在Fragment通过ViewBinding设置的文本");
return view;
}
public static ViewBindingFragment newInstance() {
Bundle args = new Bundle();
ViewBindingFragment fragment = new ViewBindingFragment();
fragment.setArguments(args);
return fragment;
}
@Override
public void onDestroyView() {
super.onDestroyView();
binding = null;
}
}
Fragment中使用注意事项
在Fragment中使用ViewBinding时,要在onDestroyView()方法中把binding的引用移除,不然会引起内存泄漏;
总结
1.使用起来比较方便,编译比较快,而且是官方推荐的;
2.目前还不支持布局变量和布局表达式;
3.不支持双向绑定,所以只适合一些简单的使用场境;
4.解决了控件空指针的问题