前言
之前已经把大部分的应用都已经介绍了,现在就把一些在概念上和应用上的细节再恶补一下。
正文
Lambda 表达式
之前我们在xml中已经掌握了如果使用variable的对象中的方法:
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:onClick="@{(v) -> listener.onClickListenerBinding(v)}"
android:text="@{user.name}" />
其中(v) -> listener.onClickListenerBinding(v) 这一句到底是什么意思呢??
其实这是引入了Lambda表达式,他能够更加简化代码,尤其是xml这种简洁的语言中就显得非常有分量。
在java中他能简化到什么地步呢,看下面这段代码:
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("Hello world !");
}
}).start();
new Thread(() -> System.out.println("Hello world !")).start();
熟悉的代码结构又出现了,跟上面的java完整代码比,确实简洁了很多,Lambda用法还有很多,我也不是非常的了解,网上有很多更为详细的资料,大家可以去仔细的研究。
xml中的多种支持
1、支持多元运算符。xml中如果绑定了bealoon型的值,难免要进行判断,DataBinding是支持多元运算符的。
<variable
name="user"
type="com.lzp.myapplication.bean.User" />
<TextView
android:text="@{user.lastName}"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:visibility="@{user.isAdult ? View.VISIBLE : View.GONE}"/>
2、支持直接使用定义了id的view。
<variable
name="user"
type="com.lzp.myapplication.bean.User" />
<import type="android.view.View" />
......
<TextView
android:id="@+id/textView"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:visibility="@{user.name != null ? View.VISIBLE : View.GONE}" />
<ImageView
android:layout_width="200dp"
android:layout_height="200dp"
android:visibility="@{testView.visibility}"/>
首先引入了android.view.View包,是为了能够使用View.VISIBLE 和View.GONE。然后id为textView的控件设置可见条件,下面的ImageView使用textView的可见度。
但是需要注意一点,这种使用方法,被依赖的view的被绑定的属性必须要使用DataBinding来进行判断,例如如果代码中的textView没有设置visibility属性,或者直接android:visibility=”gone”,这肯定就要报错了。
双向绑定
双向绑定,通俗一点的说,就是绑定的一方和被绑定的一方,任何一方发生了变化,都要更新另一方的值。
这种情况大多用在了EditText中,例如一种非常熟悉的应用场景,编辑用户的昵称和介绍,然后发送给服务器保存:
一般我们在点击事件会这么写:
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
String newName = editNameView.getText().toString();
String newDesc = editDescView.getText().toString();
...
}
});
也就是说我们要在提交时,需要手动去获取一次最新的内容,然后进行必要的验证等等操作之后再提交给服务器。
如果使用的是DataBinding:
<data>
<variable
name="newName"
type="String" />
<variable
name="newDesc"
type="String" />
</data>
...
<EditText
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="@={newName}"/>
<EditText
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="@={newDesc}"/>
...
private String newName,newDesc;
mBinder.setNewName(newName);
mBinder.setNewName(newDesc);
来分析一下上面的伪代码,首先在xml绑定了两个字符串对象,在editText中使用了这两个对象,绑定在了android:text属性上,然后在java代码中设置。
仔细的看仅仅是发现,在android:text=”@={newDesc}” 多了一个等于号(=),这就是双向绑定,在edittext输入时同时也改变了newName的值,点击时就不需要在手动获取最新的输入内容了。
如果想要自定义双向绑定就比较麻烦,个人觉得使用场景不多,而且双向绑定最注意的是避免死循环,定义的方法也比较麻烦,这样代码量反而增加了,有一种画蛇添足的感觉,具体网上也有资料,有兴趣的可以去研究一下。
DataBinding的监听器
DataBinding有两个非常重要的监听器:
1、Observable.OnPropertyChangedCallback
mBinder.addOnPropertyChangedCallback(new Observable.OnPropertyChangedCallback() {
@Override
public void onPropertyChanged(Observable observable,int i) {
}
});
这个属性是当绑定的对象发生可变化时回调的监听器,例如刚才举得TextView的例子,双向绑定的实现实际上系统利用了TextWatcher实现,所以我们就不能再使用textwatcher了,如果你想在内容发生改变时进行操作,就可以利用这个监听器,其中 int i就是 BR id。
2、OnRebindCallback
mBinder.addOnRebindCallback(new OnRebindCallback() {
@Override
public boolean onPreBind(ViewDataBinding binding) {
return super.onPreBind(binding);
}
});
这个监听器会在重新绑定的时候回调,那什么时候会重新绑定呢???
就是在layout发生了改变的时候,例如View的隐藏和消失,都会回调这个监听器。
3、上面的两个监听器,DataBinding同样提供了对应的remove方法。
总结
看到这里,DataBinding的大体我们就更加熟悉了,可能会漏掉了一些重要的内容,如果我想起来了,会及时的更新。
下一篇,就让我们来做一个简单的RecyclerView的DataBinding的例子,来结束DataBinding的学习。