DataBinding原理与编译时绑定布局与对象
2019-12-19 本文已影响0人
migill
一、编译时绑定布局与对象
1、APT预编译方式
APT(Annotation Processing Tool):
注解处理器,是一种处理注解的工具,确切的说它是javac的一个工具,它用来在编译时扫描和处理注解。注解处理器以Java代码(或者编译过的字节码)作为输入,生成.java文件作为输出。简单来说就是在编译期,通过注解生成.java文件。
2、内部处理布局文件控件
3、关联Activity组件与布局
二、DataBinding原理
1、在 Module:app的build.gradle文件添加如下代码
android {
...
// 引入DataBinding依赖
dataBinding{
enabled = true
}
}
2、创建javabean对象
public class UserInfo {
private String username;
private String password;
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
}
3、在xml文件中写DataBinding的编码规范
<?xml version="1.0" encoding="utf-8"?>
<!-- DataBinding编码规范 -->
<layout xmlns:android="http://schemas.android.com/apk/res/android">
<!-- 定义该View(布局)name数据的引用 type需要绑定的数据的绝对路径 -->
<data>
<variable
name="user"
type="com.migill.databinding.model.UserInfo" />
</data>
<!-- 布局常规编码 -->
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:gravity="center"
android:orientation="vertical">
<EditText
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@{user.username}" />
<EditText
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="30dp"
android:text="@{user.password}" />
</LinearLayout>
</layout>
这个时候就可以去重新编译项目了,可以看到会生成几个文件。
ActivityMainBindingImpl.java文件,这个是通过APT生成的文件
app/build/intermediates/incremental/mergeDebugResources/stripped.dir/layout/activity_main.xml文件
activity_main-layout.xml文件
4、Activity中书写代码绑定
1、单向绑定第一种方式:<Model-View>
public class MainActivity extends AppCompatActivity {
private UserInfo userInfo = new UserInfo();
private final static String TAG = "TAG";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ActivityMainBinding binding = DataBindingUtil.setContentView(this, R.layout.activity_main);
//单向绑定第一种方式:<Model-View>
userInfo.setUsername("migill");
userInfo.setPassword("123");
binding.setUser(userInfo);
Log.e(TAG, userInfo.getUsername() + " / " + userInfo.getPassword());
//延时3秒钟后,重新设置Model,View的显示会跟着变化吗?
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
userInfo.setUsername("pandi");
userInfo.setPassword("134");
Log.e(TAG, userInfo.getUsername() + " / " + userInfo.getPassword());
}
}, 3000);
}
}
如下图所示,更新数据后View不会跟着显示,这个是为什么呢?
MainActivity中的代码不变,修改了UserInfo中的代码,如下,可以到model数据的改变,View也跟着刷新。
2、单向绑定第二种方式:<Model-View>
import android.databinding.ObservableField;
public class UserInfo{
public ObservableField<String> username = new ObservableField<>();
public ObservableField<String> password = new ObservableField<>();
}
public class MainActivity extends AppCompatActivity {
private UserInfo userInfo = new UserInfo();
private final static String TAG = "TAG";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ActivityMainBinding binding = DataBindingUtil.setContentView(this, R.layout.activity_main);
//单向绑定第二种方式:<Model-View>
userInfo.username.set("migill");
userInfo.password.set("123");
binding.setUser(userInfo);
Log.e(TAG, userInfo.username.get() + " / " + userInfo.password.get());
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
userInfo.username.set("pandi");
userInfo.password.set("134");
Log.e(TAG, userInfo.username.get() + " / " + userInfo.password.get());
}
}, 3000);
}
}
延时3秒钟后,重新设置Model,View会跟着变化。
3、双向绑定(Model-View View-Model)
public class MainActivity extends AppCompatActivity {
private UserInfo userInfo = new UserInfo();
private final static String TAG = "TAG";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ActivityMainBinding binding = DataBindingUtil.setContentView(this, R.layout.activity_main);
//双向绑定(Model-View View-Model)
userInfo.username.set("migill");
userInfo.password.set("123");
binding.setUser(userInfo);
Log.e(TAG, userInfo.username.get() + " / " + userInfo.password.get());
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
Log.e(TAG, userInfo.username.get() + " / " + userInfo.password.get());
}
},20000);
}
}
如下图,可以看出,View的变化,Model没有跟着改变,这个是为什么呢?
原因:
Model是如何刷新View
Model是如何刷新View
总结:使用了databinding的项目,随着项目的越来越大,内存的销毁也会越来越大,编译时间越来越长,它会不断的产生新的class文件出来。