双向绑定涉及两个部分,即将业务状态的变化传递给UI,以及将用户输入信息传递给业务模型。java
首先咱们来看业务状态是如何传递给UI的。开启dataBinding后,编译器为布局activitymain.xml生成类ActivityMainBinding和它的实现类ActivityMainBindingImpl。抽象类ActivityMainBinding派生自ViewDataBinding,后者是BaseObserver的子类。ViewDataBinding有一个成员mChoreographer,它保存了主线程ActivityThread的Looper对象,业务状态的变动就是经过消息发送给主线程的,接着主线程完成对UI的更新。咱们以一个例子来讲明。对于下面的布局:android
<?xml version="1.0" encoding="utf-8"?> <layout> <data> <variable name="text" type="java.lang.String"/> </data> <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" xmlns:app="http://schemas.android.com/apk/res-auto" android:layout_width="match_parent" android:layout_height="match_parent" tools:context=".MainActivity"> <EditText android:id="@+id/label" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="@={text}" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintLeft_toLeftOf="parent" app:layout_constraintRight_toRightOf="parent" app:layout_constraintTop_toTopOf="parent"/> </androidx.constraintlayout.widget.ConstraintLayout> </layout>
当调用ActivityMainBinding.setText()时,执行的操做大体以下:app
ActivityMainBindingImpl.setText() ViewDataBinding.requestRebind() ViewDataBinding.mChoregrapher.postFrameCallback() some_native_code() Choregrapher.FrameDisplayEventReceiver.onVsync() Choregrapher.FrameDisplayEventReceiver.mHandler.sendMessage() ... Handler.dispatchMessage() Choregrapher.FrameDisplayEventReceiver.run() Choregrapher.doFrame() ViewDataBinding.executeBindings() TextViewBindingAdapter.setText() TestView.setText()
这里有两个地方要注意一下。首先,Choregrapher经过Looper.myLooper()获得Looper对象,而myLooper()返回了一个线程本地对象,所以ViewDataBinding对象必须在主线程中建立。其次,Handler分派消息(Handler.dispatchMessage)的顺序为Message.callback,Handler.mCallback,Handler.handleMessage。这种分派方式提升了消息处理的灵活性。oop
如今来考察用户输入是如何传递给业务模型的。当用户输入数据时,EditText会将信息传递给TextWatcher。双向绑定会自动创建TextWatcher对象,并注册到EditText,这个TextWatcher包含一个InverseBindingListener对象,当用户输入时,信息沿着下面的路径传递给ActivityMainBindingImpl对象。布局
EditText TextWatcher InverseBindingListener ActivityMainBindingImpl
所以经过ActivityMainBindingImpl对象,咱们就能够直接获得用户的输入了。post