先说结论
ObservableField 一般用于 dataBinding 交互式绑定。
ObservableField.set() 从时效性来说并不等同于 TextView.setText()。
解决办法,如果需要实时处理观察结果 ,通过
Choreographer.getInstance().postFrameCallback() 处理。
举个简单例子
public class TestViewModel extends ViewModel {
public ObservableField<String> text = new ObservableField<>();
public void setText(String s) {
text.set(s);
}
}
public class TestActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); View view = getLayoutInflater().inflate(R.layout.activity_test, null); setContentView(view); DataBindingUtil.bind(view); ActivityTestBinding binding = DataBindingUtil.getBinding(view); TestViewModel viewModel = new ViewModelProvider(this, new ViewModelProvider.Factory() { @NonNull @Override public <T extends ViewModel> T create(@NonNull Class<T> modelClass) { return (T) new TestViewModel(); } }).get(TestViewModel.class); binding.setViewModel(viewModel); binding.btTest.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { viewModel.setText("天气真好"); Log.e("TEST", "text: " + binding.tvTest.getText()); } }); } }
查看ui,可以看到没有问题正常渲染,那么所说的延迟是体现在哪里呢。查看日志
可以看到 viewModel.setText("天气真好") 之后并没有立即调用TestView.setText。通过对源码的追踪分析找到问题所在。
对于Choreographer感兴趣的朋友可以去Android 官网看一下,
mChoreographer.postFrameCallback的作用是 渲染下一帧时执行此回调,那么到这里就很清楚了,想要ObservableField.set 之后立马处理响应结果,可以这样写。
binding.btTest.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
viewModel.setText("天气真好");
Choreographer.getInstance().postFrameCallback(new Choreographer.FrameCallback() {
@Override
public void doFrame(long l) {
Log.e("TEST", "text: " + binding.tvTest.getText());
}
});
}
});
下面是发现问题的业务场景 。
结合ObservableField 与dataBinding,RecyclerView绑定Adapteer,此处使用的是开源框架
BRVAH。
ECGDataAdapter ecgDataAdapter = new ECGDataAdapter(list);
mAdapter.set(ecgDataAdapter);
ecgDataAdapter.setEmptyView(R.layout.iv_placeholder);
查看setEmptyView()函数,此处因为 mAdapter.set(ecgDataAdapter) 并没有立即发送事件,也就是 recyclerView.setAdapter 进行绑定,导致adapter .mRecyclerView为null,所以设置EmptyView无效。
修改为
ECGDataAdapter ecgDataAdapter = new ECGDataAdapter(list);
mAdapter.set(ecgDataAdapter);
Choreographer.getInstance().postFrameCallback(frameTimeNanos -> ecgDataAdapter.setEmptyView(R.layout.iv_placeholder));