前言

DataBinding 数据绑定库是 Android Jetpack 的一部分,借助该库可以使用声明性格式(而非程序化地)将布局中的界面组件绑定到应用中的数据源。我个人觉得,使用 DataBinding 时不要在 xml 布局文件中写复杂的逻辑,只负责绑定数据。只是负责将最终的数据和 UI 直接绑定,只是一个末端赋值而已,不涉及复杂的 UI 逻辑,而且避免了代码中大量冗余代码的判空处理,同时避免了那些常见的 setVisible 等样板方法的调用,简化开发流程,统一 UI 的数据来源。

基本使用

简单使用入门

xml布局如下:

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <layout xmlns:android="http://schemas.android.com/apk/res/android"
  3. xmlns:app="http://schemas.android.com/apk/res-auto"
  4. xmlns:tools="http://schemas.android.com/tools">
  5. <data>
  6. <import type="com.jackie.jetpackdemo.data.TestInfo"/>
  7. <variable
  8. name="userInfo"
  9. type="TestInfo" />
  10. </data>
  11. <androidx.constraintlayout.widget.ConstraintLayout
  12. android:layout_width="match_parent"
  13. android:layout_height="match_parent"
  14. tools:context=".MainActivity">
  15. <Button
  16. android:id="@+id/btnGetUserInfo"
  17. android:layout_width="wrap_content"
  18. android:layout_height="wrap_content"
  19. android:text="获取用户信息"
  20. app:layout_constraintBottom_toBottomOf="parent"
  21. app:layout_constraintLeft_toLeftOf="parent"
  22. app:layout_constraintRight_toRightOf="parent"
  23. app:layout_constraintTop_toTopOf="parent" />
  24. <TextView
  25. android:id="@+id/txtUserName"
  26. android:layout_width="match_parent"
  27. android:layout_height="wrap_content"
  28. android:gravity="center"
  29. android:text="@{userInfo.age}"
  30. app:layout_constraintTop_toBottomOf="@+id/btnGetUserInfo"
  31. android:layout_marginTop="30dp"
  32. android:textSize="30dp"
  33. />
  34. <TextView
  35. android:layout_width="match_parent"
  36. android:layout_height="wrap_content"
  37. android:text="text"
  38. android:gravity="center"
  39. app:layout_constraintTop_toBottomOf="@+id/txtUserName"
  40. />
  41. </androidx.constraintlayout.widget.ConstraintLayout>
  42. </layout>

Activity 中调用代码如下:

  1. override fun onCreate(savedInstanceState: Bundle?) {
  2. super.onCreate(savedInstanceState)
  3. val activityBinding: ActivityMainBinding = DataBindingUtil.setContentView(this,R.layout.activity_main)
  4. activityBinding.lifecycleOwner = this
  5. activityBinding.userInfo = TestInfo("lsm","lsj")
  6. }

TestInfo 的定义如下:

  1. public class TestInfo extends BaseObservable { //继承 BaseObservable
  2. private String age;
  3. private String name;
  4. public TestInfo(String age,String name){
  5. this.name = name;
  6. this.age = age;
  7. }
  8. public void setAge(String age) {
  9. this.age = age;
  10. notifyPropertyChanged(BR.age); //需要变更的变量的 set 方法中加上 notifyPropertyChanged
  11. }
  12. public void setName(String name) {
  13. this.name = name;
  14. notifyPropertyChanged(BR.name); //需要变更的变量的 set 方法中加上 notifyPropertyChanged
  15. }
  16. @Bindable //需要变更的变量还要加上 @Bindable 注解
  17. public String getAge() {
  18. return age;
  19. }
  20. @Bindable //需要变更的变量还要加上 @Bindable 注解
  21. public String getName() {
  22. return name;
  23. }
  24. }

TestInfo需要继承BaseObservable,同时对于需要监听变化的变量加上@Bindable 注解,同时该变量的set方法还要加上notifyPropertyChangedBR.xxx 是注解生成的。

数据的双向绑定

使用单向数据绑定时,您可以为特性设置值,并设置对该特性的变化作出反应的监听器:

  1. <CheckBox
  2. android:id="@+id/rememberMeCheckBox"
  3. android:checked="@{viewmodel.rememberMe}"
  4. android:onCheckedChanged="@{viewmodel.rememberMeChanged}"
  5. />

双向数据绑定为此过程提供了一种快捷方式:

  1. <CheckBox
  2. android:id="@+id/rememberMeCheckBox"
  3. android:checked="@={viewmodel.rememberMe}"
  4. />

@={} 表示法(其中重要的是包含“=”符号)可接收属性的数据更改并同时监听用户更新。其他的设置和前面的单向数据绑定一致。

结合 LiveData 使用

内容参考自这里,我们上面在使用 DataBinding 时,TestInfo 还要继承 BaseObserble,使用注解、notifyPropertyChanged(),使用起来其实挺复杂,而且还有侵入性。LiveData 结合 DataBinding 的使用步骤如下:

  1. 使用 LiveData 对象作为数据绑定来源,需要设置 LifecycleOwner。
  2. xml 中定义变量 ViewModel,并使用 ViewModel。
  3. binding 设置变量 ViewModel。
  1. //结合DataBinding使用的ViewModel
  2. //1\. 要使用LiveData对象作为数据绑定来源,需要设置LifecycleOwner
  3. binding.setLifecycleOwner(this);
  4. ViewModelProvider viewModelProvider = new ViewModelProvider(this);
  5. mUserViewModel = viewModelProvider.get(UserViewModel.class);
  6. //3\. 设置变量ViewModel
  7. binding.setVm(mUserViewModel);

xml 文件的定义如下:

  1. <!-- 2\. 定义ViewModel 并绑定-->
  2. <variable
  3. name="vm"
  4. type="com.hfy.demo01.module.jetpack.databinding.UserViewModel" />
  5. <TextView
  6. android:layout_width="wrap_content"
  7. android:layout_height="wrap_content"
  8. android:text="@{vm.userLiveData.name}"/>
  9. <TextView
  10. android:layout_width="wrap_content"
  11. android:layout_height="wrap_content"
  12. android:text="@{vm.userLiveData.level}"/>

这样就ok了,你会发现 我们不需要在 Activity 中拿到 LivaData 去 observe(owner,observer)了,DataBinding 自动生成的代码,会帮我们去做这操作,所以需要设置LifecycleOwner。

使用自定义特性的双向数据绑定

例如,如果要在名为 MyView 的自定义视图中对 "time" 特性启用双向数据绑定,请完成以下步骤:

  1. 使用 @BindingAdapter,对用来设置初始值并在值更改时进行更新的方法进行注释:
  1. @BindingAdapter("time")
  2. @JvmStatic fun setTime(view: MyView, newValue: Time) {
  3. // Important to break potential infinite loops.
  4. if (view.time != newValue) {
  5. view.time = newValue
  6. }
  7. }
  1. 使用 @InverseBindingAdapter 对从视图中读取值的方法进行注释:
  1. @InverseBindingAdapter("time")
  2. @JvmStatic fun getTime(view: MyView) : Time {
  3. return view.getTime()
  4. }

更多内容请参考这里

源码分析

我们在路径app/build/intermediates/data_binding_layout_info_type_merge/debug/out/activity_main-layout.xml查看文件

  1. <?xml version="1.0" encoding="utf-8" standalone="yes"?>
  2. <Layout directory="layout" filePath="app/src/main/res/layout/activity_main.xml"
  3. isBindingData="true" isMerge="false" layout="activity_main"
  4. modulePackage="com.jackie.jetpackdemo" rootNodeType="androidx.constraintlayout.widget.ConstraintLayout">
  5. <Variables name="userInfo" declared="true" type="TestInfo">
  6. <location endLine="9" endOffset="29" startLine="7" startOffset="8" />
  7. </Variables>
  8. <Imports name="TestInfo" type="com.jackie.jetpackdemo.data.TestInfo">
  9. <location endLine="6" endOffset="60" startLine="6" startOffset="8" />
  10. </Imports>
  11. <Targets>
  12. <Target tag="layout/activity_main_0"
  13. view="androidx.constraintlayout.widget.ConstraintLayout">
  14. <Expressions />
  15. <location endLine="47" endOffset="55" startLine="12" startOffset="4" />
  16. </Target>
  17. <Target id="@+id/txtUserName" tag="binding_1" view="TextView">
  18. <Expressions>
  19. <Expression attribute="android:text" text="userInfo.age">
  20. <Location endLine="32" endOffset="41" startLine="32" startOffset="12" />
  21. <TwoWay>false</TwoWay>
  22. <ValueLocation endLine="32" endOffset="39" startLine="32" startOffset="28" />
  23. </Expression>
  24. </Expressions>
  25. <location endLine="37" endOffset="13" startLine="27" startOffset="8" />
  26. </Target>
  27. <Target id="@+id/btnGetUserInfo" view="Button">
  28. <Expressions />
  29. <location endLine="25" endOffset="55" startLine="17" startOffset="8" />
  30. </Target>
  31. </Targets>
  32. </Layout>

可以看到<Targets>标签下面的就是我们布局,分成具体的子<Target>标签对应具体的 ConstraintLayout、TextView等,activity_main_0 对应我们的ConstraintLayout,再来路径app/build/intermediates/incremental/mergeDebugResources/stripped.dir/layout/activity_main.xml

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <androidx.constraintlayout.widget.ConstraintLayout
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent"
  5. tools:context=".MainActivity" android:tag="layout/activity_main_0" 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">
  6. <Button
  7. android:id="@+id/btnGetUserInfo"
  8. android:layout_width="wrap_content"
  9. android:layout_height="wrap_content"
  10. android:text="获取用户信息"
  11. app:layout_constraintBottom_toBottomOf="parent"
  12. app:layout_constraintLeft_toLeftOf="parent"
  13. app:layout_constraintRight_toRightOf="parent"
  14. app:layout_constraintTop_toTopOf="parent" />
  15. <TextView
  16. android:id="@+id/txtUserName"
  17. android:layout_width="match_parent"
  18. android:layout_height="wrap_content"
  19. android:gravity="center"
  20. android:tag="binding_1"
  21. app:layout_constraintTop_toBottomOf="@+id/btnGetUserInfo"
  22. android:layout_marginTop="30dp"
  23. android:textSize="30dp"
  24. />
  25. <TextView
  26. android:layout_width="match_parent"
  27. android:layout_height="wrap_content"
  28. android:text="text"
  29. android:gravity="center"
  30. app:layout_constraintTop_toBottomOf="@+id/txtUserName"
  31. />
  32. </androidx.constraintlayout.widget.ConstraintLayout>

这个文件其实就是移除了<layout>、<data>标签的布局文件,里面的 tag 就是我们上面对应<Target>标签中的tagExpression attribute="android:text" text="userInfo.age" <Expression>中的具体属性对应具体的值。

初始化

再来从val activityBinding: ActivityMainBinding = DataBindingUtil.setContentView(this,R.layout.activity_main)来分析源码,

  1. //DataBindingUtil.java
  2. public static <T extends ViewDataBinding> T setContentView(@NonNull Activity activity,
  3. int layoutId, @Nullable DataBindingComponent bindingComponent) {
  4. activity.setContentView(layoutId); //还是需要setContentView
  5. View decorView = activity.getWindow().getDecorView();
  6. ViewGroup contentView = (ViewGroup) decorView.findViewById(android.R.id.content);
  7. return bindToAddedViews(bindingComponent, contentView, 0, layoutId);
  8. }

我们设置的activity_xxx.xml其实是在android.R.id.content下面的,继续来看bindToAddedViews方法

  1. //DataBindingUtil.java
  2. private static <T extends ViewDataBinding> T bindToAddedViews(DataBindingComponent component,
  3. ViewGroup parent, int startChildren, int layoutId) {
  4. final int endChildren = parent.getChildCount();
  5. final int childrenAdded = endChildren - startChildren;
  6. if (childrenAdded == 1) {
  7. final View childView = parent.getChildAt(endChildren - 1);
  8. return bind(component, childView, layoutId); //调用bind
  9. } else {
  10. final View[] children = new View[childrenAdded];
  11. for (int i = 0; i < childrenAdded; i++) {
  12. children[i] = parent.getChildAt(i + startChildren);
  13. }
  14. return bind(component, children, layoutId); //调用bind
  15. }
  16. }
  17. }

最终会调到bind方法

  1. //DataBindingUtil.java
  2. static <T extends ViewDataBinding> T bind(DataBindingComponent bindingComponent, View root,
  3. int layoutId) {
  4. return (T) sMapper.getDataBinder(bindingComponent, root, layoutId);
  5. }

sMapperDataBinderMapper,其真正实现类是通过 APT 生成的DataBinderMapperImpl(app/build/generated/ap_generated_sources/debug/out/com/jackie/jetpackdemo/DataBinderMapperImpl.java)

  1. public class DataBinderMapperImpl extends DataBinderMapper {
  2. ···
  3. @Override
  4. public ViewDataBinding getDataBinder(DataBindingComponent component, View view, int layoutId) {
  5. int localizedLayoutId = INTERNAL_LAYOUT_ID_LOOKUP.get(layoutId);
  6. if(localizedLayoutId > 0) {
  7. final Object tag = view.getTag();
  8. if(tag == null) {
  9. throw new RuntimeException("view must have a tag");
  10. }
  11. switch(localizedLayoutId) {
  12. case LAYOUT_ACTIVITYMAIN: {
  13. if ("layout/activity_main_0".equals(tag)) {
  14. return new ActivityMainBindingImpl(component, view); //关键代码,new ActivityMainBindingImpl
  15. }
  16. ···

接下来我们来分析ActivityMainBindingImpl(app/build/generated/ap_generated_sources/debug/out/com/jackie/jetpackdemo/databinding/ActivityMainBindingImpl.java)这个类,它也是 APT 生成的,

  1. //ActivityMainBindingImpl.java
  2. public ActivityMainBindingImpl(@Nullable androidx.databinding.DataBindingComponent bindingComponent, @NonNull View root) {
  3. this(bindingComponent, root, mapBindings(bindingComponent, root, 3, sIncludes, sViewsWithIds));
  4. }
  5. private ActivityMainBindingImpl(androidx.databinding.DataBindingComponent bindingComponent, View root, Object[] bindings) {
  6. super(bindingComponent, root, 1
  7. , (android.widget.Button) bindings[2]
  8. , (android.widget.TextView) bindings[1]
  9. );
  10. this.mboundView0 = (androidx.constraintlayout.widget.ConstraintLayout) bindings[0];
  11. this.mboundView0.setTag(null);
  12. this.txtUserName.setTag(null);
  13. setRootTag(root);
  14. // listeners
  15. invalidateAll();
  16. }

我们调用了第一个方法,里面的这个 3 代表着我们布局文件中有三个节点(ConstraintLayout,Button,TextView),但是我们前面的布局中明明还有一个TextView,为什么没有呢?因为我们这个TextView我们并没有设置它的 Id,所以没有生成,如果设置后重新 build 下 3 就会变成 4 了。

继续来看mapBindings方法:

  1. protected static Object[] mapBindings(DataBindingComponent bindingComponent, View root,
  2. int numBindings, IncludedLayouts includes, SparseIntArray viewsWithIds) {
  3. Object[] bindings = new Object[numBindings];
  4. mapBindings(bindingComponent, root, bindings, includes, viewsWithIds, true);
  5. return bindings;
  6. }

它首先是 new 一个大小为 3 的对象数组,然后把这三个标签解析完放到该数组中。上面的ActivityMainBindingImpl公有构造器会调用私有构造器,再回过头来看

  1. val activityBinding: ActivityMainBinding = DataBindingUtil.setContentView(this,R.layout.activity_main)

执行完这个代码,activityBinding中已经有这 3 个对象了,所以可以进行这样的调用

  1. activityBinding.txtUserName
  2. activityBinding.btnGetUserInfo

到目前为止,初始化已经完成了。

调用流程

接下来我们从这个调用开始分析

  1. activityBinding.userInfo = TestInfo("lsm","lsj")

这个activityBinding.userInfo调用的实际上是ActivityMainBinding中的setUserInfo方法

  1. //ActivityMainBinding.java
  2. public void setUserInfo(@Nullable com.jackie.jetpackdemo.data.TestInfo UserInfo) {
  3. updateRegistration(0, UserInfo);
  4. this.mUserInfo = UserInfo;
  5. synchronized(this) {
  6. mDirtyFlags |= 0x1L;
  7. }
  8. notifyPropertyChanged(BR.userInfo);
  9. super.requestRebind();
  10. }

这里的updateRegistration方法如下

  1. //localFieldId 为 BR 文件中的Id,observable 就是观察者
  2. protected boolean updateRegistration(int localFieldId, Observable observable) {
  3. return updateRegistration(localFieldId, observable, CREATE_PROPERTY_LISTENER);
  4. }
  5. /**
  6. * Method object extracted out to attach a listener to a bound Observable object.
  7. */
  8. private static final CreateWeakListener CREATE_PROPERTY_LISTENER = new CreateWeakListener() {
  9. @Override
  10. public WeakListener create(ViewDataBinding viewDataBinding, int localFieldId) {
  11. return new WeakPropertyListener(viewDataBinding, localFieldId).getListener();
  12. }
  13. };

CREATE_PROPERTY_LISTENER名字也很直白,表示创建一个属性的监听器,也就是说属性发生变化的时候WeakPropertyListener监听器会被回调。

localFieldId 为 BR 文件中的 Id,BR 文件是什么呢?

  1. public class BR {
  2. public static final int _all = 0;
  3. public static final int age = 1;
  4. public static final int name = 2;
  5. public static final int userInfo = 3;
  6. }

因为我们的 xml 文件中导入了TestInfo(userInfo),我们也在agename属性上加上了@Bindable注解,所以生成了上面的BR文件。因为我们是上面调用的是setUserInfo方法,所以传入的是 0。

用该方式设置name也可以

  1. activityBinding.setVariable(BR.name,"Jackie")

updateRegistration中的observable就是我们传入的(TestInfo)UserInfo,再来看看updateRegistration方法

  1. //ViewDataBinding
  2. private boolean updateRegistration(int localFieldId, Object observable,
  3. CreateWeakListener listenerCreator) {
  4. if (observable == null) {
  5. return unregisterFrom(localFieldId);
  6. }
  7. WeakListener listener = mLocalFieldObservers[localFieldId];
  8. if (listener == null) {
  9. registerTo(localFieldId, observable, listenerCreator);
  10. return true;
  11. }
  12. if (listener.getTarget() == observable) {
  13. return false;//nothing to do, same object
  14. }
  15. unregisterFrom(localFieldId);
  16. registerTo(localFieldId, observable, listenerCreator);
  17. return true;
  18. }

mLocalFieldObservers数组中绑定了每一个属性对应的监听器,比如我们上面的 BR 中的四个值。

如果监听器为空,调用registerTo创建监听器并注册

  1. protected void registerTo(int localFieldId, Object observable,
  2. CreateWeakListener listenerCreator) {
  3. if (observable == null) {
  4. return;
  5. }
  6. WeakListener listener = mLocalFieldObservers[localFieldId];
  7. if (listener == null) {
  8. listener = listenerCreator.create(this, localFieldId);
  9. mLocalFieldObservers[localFieldId] = listener;
  10. if (mLifecycleOwner != null) {
  11. listener.setLifecycleOwner(mLifecycleOwner);
  12. }
  13. }
  14. listener.setTarget(observable);
  15. }

setTarget就是给被观察者添加监听器

  1. public void setTarget(T object) {
  2. unregister();
  3. mTarget = object;
  4. if (mTarget != null) {
  5. mObservable.addListener(mTarget);
  6. }
  7. }

这里的mObservable的实现类是WeakPropertyListener,也就是每个属性发生变化后都会进行回调

  1. @Override
  2. public void addListener(Observable target) {
  3. target.addOnPropertyChangedCallback(this);
  4. }

target的实现类是BaseObservable,这也就是我们的TestInfo为什么要继承BaseObservable了。

  1. public class BaseObservable implements Observable {
  2. private transient PropertyChangeRegistry mCallbacks;
  3. public BaseObservable() {
  4. }
  5. @Override
  6. public void addOnPropertyChangedCallback(@NonNull OnPropertyChangedCallback callback) {
  7. synchronized (this) {
  8. if (mCallbacks == null) {
  9. mCallbacks = new PropertyChangeRegistry();
  10. }
  11. }
  12. mCallbacks.add(callback);
  13. }

总的关系图如下:

PropertryChangeRegistry中的add(ViewDataBinding)是将观察者和被观察者绑定起来,ViewDataBinding中的WeakListener[] mLocalFieldObservers中的每个变量都有一个WeakListenerBaseObservable中的addOnPropertyChangedCallback(WeakPropertyListener)就是添加属性变化回调。

MainActivity 调用 setUserInfo 流程图

如果上面的关系图不清楚的话,我也把流程图画出来,你可以查看一下

ActivityMainBindingImpl继承于ActivityMainBinding,而ActivityMainBinding继承于ViewDataBinding

最终是在TextViewBindingAdaptersetText来实现。

  1. @BindingAdapter("android:text")
  2. public static void setText(TextView view, CharSequence text) {
  3. final CharSequence oldText = view.getText();
  4. if (text == oldText || (text == null && oldText.length() == 0)) {
  5. return;
  6. }
  7. if (text instanceof Spanned) {
  8. if (text.equals(oldText)) {
  9. return; // No change in the spans, so don't set anything.
  10. }
  11. } else if (!haveContentsChanged(text, oldText)) {
  12. return; // No content changes, so don't set anything.
  13. }
  14. view.setText(text);
  15. }
没调用 setUser 之前的数据绑定是怎么做的?

前面我们讲了初始化过程,因为ActivityMainBindingImpl继承于ActivityMainBinding,而ActivityMainBinding继承于ViewDataBinding,在ViewDataBinding中静态初始化块如下

  1. static {
  2. if (VERSION.SDK_INT < VERSION_CODES.KITKAT) {
  3. ROOT_REATTACHED_LISTENER = null;
  4. } else {
  5. ROOT_REATTACHED_LISTENER = new OnAttachStateChangeListener() {
  6. @TargetApi(VERSION_CODES.KITKAT)
  7. @Override
  8. public void onViewAttachedToWindow(View v) {
  9. // execute the pending bindings.
  10. final ViewDataBinding binding = getBinding(v);
  11. binding.mRebindRunnable.run();
  12. v.removeOnAttachStateChangeListener(this);
  13. }
  14. @Override
  15. public void onViewDetachedFromWindow(View v) {
  16. }
  17. };
  18. }
  19. }

会执行mRebindRunnablerun方法

  1. private final Runnable mRebindRunnable = new Runnable() {
  2. @Override
  3. public void run() {
  4. synchronized (this) {
  5. mPendingRebind = false;
  6. }
  7. processReferenceQueue();
  8. if (VERSION.SDK_INT >= VERSION_CODES.KITKAT) {
  9. // Nested so that we don't get a lint warning in IntelliJ
  10. if (!mRoot.isAttachedToWindow()) {
  11. // Don't execute the pending bindings until the View
  12. // is attached again.
  13. mRoot.removeOnAttachStateChangeListener(ROOT_REATTACHED_LISTENER);
  14. mRoot.addOnAttachStateChangeListener(ROOT_REATTACHED_LISTENER);
  15. return;
  16. }
  17. }
  18. executePendingBindings();
  19. }
  20. };

然后执行executePendingBindings方法,后面就和我们前面分析过的的流程一样了。最终会会调到

  1. @BindingAdapter("android:text")
  2. public static void setText(TextView view, CharSequence text) {
  3. final CharSequence oldText = view.getText();
  4. if (text == oldText || (text == null && oldText.length() == 0)) {
  5. return;
  6. }
  7. if (text instanceof Spanned) {
  8. if (text.equals(oldText)) {
  9. return; // No change in the spans, so don't set anything.
  10. }
  11. } else if (!haveContentsChanged(text, oldText)) {
  12. return; // No content changes, so don't set anything.
  13. }
  14. view.setText(text);
  15. }

我们一开始text为空,所以就直接返回了。

为什么设计如此复杂,以前我们用的话观察者和被观察者都是单独的类,现在因为可能有多个ViewModel(TestInfo、xxxInfo等),同时也可能有多个Activity(分别进行绑定),使用原来的方式消耗很大。BR 文件会生成多个字段,用一个mLocalFieldObservers进行处理,每个字段有其具体的监听,每个xxxInfo 有其具体的监听。设计的还是挺合理的。

总结

本文先从 DataBinding 的简单使用,单向/双向绑定,结合LiveData 使用,同时也可以使用一些自定义特性,最后是进行源码分析。最后一定要注意的是不要在 xml 中做复杂的逻辑判断,只是将其看做是一个方便末端 UI 展示的支持库即可,同时也避免了大量的判空处理,同时也统一了数据的来源。

献给读者

随着互联网企业的不断发展,产品项目中的模块越来越多,用户体验要求也越来越高,想实现小步快跑、快速迭代的目的越来越难,还有65535,应用之间的互相调用等等问题,插件化技术应用而生。如果没有插件化技术,美团、淘宝这些集成了大量“app”的应用,可能会有几个g那么大。

所以,当今的Android移动开发,不会热修复、插件化、组件化,80%以上的面试都过不了。

Android热修复框架、插件化框架、组件化框架、图片加载框架、网络访问框架、RxJava响应式编程框架、IOC依赖注入框架、最近架构组件Jetpack等等Android开源框架。系统教程知识笔记已整理成PDF电子书上传在【GitHub】

1042页完整版PDF点击我就可以白嫖啦,记得给文章点个赞哦。

文末

感谢大家关注我,分享Android干货,交流Android技术。

对文章有何见解,或者有何技术问题,都可以在评论区一起留言讨论,我会虔诚为你解答。

也欢迎大家来我的B站找我玩,有各类Android架构师进阶技术难点的视频讲解,助你早日升职加薪。

B站直通车:https://space.bilibili.com/544650554

【Android Jetpack高手日志】DataBinding 从入门到精通的更多相关文章

  1. Android开发书籍推荐:从入门到精通系列学习路线书籍介绍

    Android开发书籍推荐:从入门到精通系列学习路线书籍介绍 很多时候我们都会不断收到新手的提问"Android开发的经典入门教材和学习路线?"."Android 开发入 ...

  2. (转)Android开发书籍推荐:从入门到精通系列学习路线书籍介绍

    Android开发书籍推荐:从入门到精通系列学习路线书籍介绍 转自:http://blog.csdn.net/findsafety/article/details/52317506 很多时候我们都会不 ...

  3. Android OpenGL ES 开发教程 从入门到精通

    感谢,摘自:http://blog.csdn.net/mapdigit/article/details/7526556 Android OpenGL ES 简明开发教程 Android OpenGL ...

  4. 学习Android Jetpack? 入门教程和进阶实战这里全都有!

    前言 2018年谷歌I/O,Jetpack横空出世,官方介绍如下: Jetpack 是一套库.工具和指南,可帮助开发者更轻松地编写优质应用.这些组件可帮助您遵循最佳做法.让您摆脱编写样板代码的工作并简 ...

  5. Android Jetpack从入门到精通(深度好文,值得收藏)

    前言 即学即用Android Jetpack系列Blog的目的是通过学习Android Jetpack完成一个简单的Demo,本文是即学即用Android Jetpack系列Blog的第一篇. 记得去 ...

  6. Android官方数据绑定框架DataBinding

    数据绑定框架给我们带来了更大的方便性,以前我们可能需要在Activity里写很多的findViewById,烦人的代码也增加了我们代码的耦合性,现在我们马上就可以抛弃那么多的findViewById. ...

  7. Android数据库高手秘籍(一)——SQLite命令

    转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/38461239 要想熟练地操作不论什么一个数据库.最最主要的要求就是要懂SQL语言, ...

  8. Android Studio2.0 教程从入门到精通Windows版 - 入门篇

    http://www.open-open.com/lib/view/open1468121363300.html 本文转自:深度开源(open-open.com)原文标题:Android Studio ...

  9. Android开发高手课NOTE

    最近学习了极客时间的<Android开发高手课>很有收获,记录总结一下. 欢迎学习老师的专栏:Android开发高手课 内存优化 卡顿的原因 频繁 GC 造成卡顿.物理内存不足时系统会触发 ...

随机推荐

  1. java关于字符串是否存

    1, if('true'.equalsIgnoreCase(response.result as String)); 2,   if (scvrsp.toLowerCase().contains(&q ...

  2. 超详细Linux新手快速入门(一)——Linux的介绍安装以及虚拟机的介绍安装

    一.Linux的介绍 1.Linux和Windows的比较  Linux是一款操作系统,其性能稳定,因其防火墙组件高效安全.简单易配置,所以获得了追求速度和安全的一些企业和人群的青睐.与我们日常所熟知 ...

  3. js输入框只能输入数字

    1.只允许输入数字 <input type="text"  onkeyup="this.value=this.value.replace(/\D/g,'')&quo ...

  4. Solon 框架详解(十)- Solon 的常用配置

    Springboot min -Solon 详解系列文章: Springboot mini - Solon详解(一)- 快速入门 Springboot mini - Solon详解(二)- Solon ...

  5. get和post的区别主要有以下几方面

    1.url可见性: get,参数url可见: post,url参数不可见 2.数据传输上: get,通过拼接url进行传递参数: post,通过body体传输参数 3.缓存性: get请求是可以缓存的 ...

  6. gtk+2.0中函数set_widget_font_size()函数在编译时未定义的解决办法

    自己写一个头文件即可,代码如下: 在.c文件中包含该头文件即可

  7. Docker 专题总结

    Docker 专题总结 Docker 的基本命令 启动 Docker $ systemctl start docker 停止Docker $ systemctl stop docker 重启Docke ...

  8. Spring基于XML的IoC

    Maven导入Spring依赖 <dependency> <groupId>org.springframework</groupId> <artifactId ...

  9. java例题_49 计算子串出现的次数

    1 /*49 [程序 49 子串出现的个数] 2 题目:计算字符串中子串出现的次数 3 */ 4 5 /*分析 6 * 1.子串的出现是有标志的,如" ",*,#或者其他 7 * ...

  10. 用 customRef 做一个防抖函数,支持 element 等UI库。

    这几天学习Vue的官网,看到 customRef 提供了一个例子,研究半天发现这是一个防抖函数,觉得挺好,于是把这个例子扩展了一下,可以用于表单子控件和查询子控件. 需求 v-model 基于 ele ...