我尝试通过数据绑定来理解mvvm arhitecture。下面我写一个简单的例子:在片段开始时从网络加载数据。基本上,它有效。
但我有一些问题。
1(当方向改变时,片段从网络重新加载数据,因为loadFromNetwork()
onViewCreated
调用。ViewModel没有被重新加载,那么有什么正确的解决方案可以避免重新加载数据呢?我应该在ViewModel中实现"缓存",还是在View(Fragment(中进行检查?
2(当Ui 销毁时,我们需要处理网络操作。将这些清洁放置在哪里(注意:subscribe
方法返回Disposable
(?
fragment_my.xml
<layout 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">
<data>
<import type="android.view.View" />
<import type="com.xxx.State" />
<variable
name="viewModel"
type="com.xxx.MyViewModel" />
</data>
<FrameLayout
android:layout_width="match_parent"
android:layout_height="match_parent">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@{viewModel.myPojo.data}" />
<ProgressBar
android:layout_width="wrap_content"
android:layout_height="match_parent"
android:layout_gravity="center"
android:visibility="@{viewModel.state == State.LOADING ? View.VISIBLE : View.GONE}" />
</FrameLayout>
</layout>
MyViewModel.kt
class MyViewModel() : ViewModel() {
val state = ObservableField(State.LOADING)
val myPojo = ObservableField<MyPojo>()
@Inject
lateinit var netApi: NetworkApi
init {
Injector.getAppComponent().inject(this)
}
fun loadFromNetwork() {
state.set(State.LOADING)
netApi.getMyPojo().subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe({
state.set(State.SUCCESS)
myPojo.set(it)
}, {
state.set(State.ERROR)
})
}
}
MyFragment.kt
class MyFragment : Fragment(){
private lateinit var mBinding: FragmentMyBinding
private lateinit var mViewModel: MyViewModel
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
mViewModel = ViewModelProviders.of(this).get(MyViewModel::class.java)
}
override fun onCreateView(inflater: LayoutInflater?, container: ViewGroup?, savedInstanceState: Bundle?): View? {
mBinding = DataBindingUtil.inflate(inflater, R.layout.fragment_my, container, false)
return mBinding.root
}
override fun onViewCreated(view: View?, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
mBinding.viewModel = mViewModel
mViewModel.loadFromNetwork()
}
}
您可以将数据保存到首选项或 sqlite 数据库中,以避免每次都从网络获取数据。 假设你不想这样做,我可以想到两个选择。 首先,将数据存储在saveInstanceState
onDestroy
中的捆绑包并在onCreate
中读回的标准方法。 其次,如果您有一个 singleTop"主"活动或片段,我已经成功地将 ViewModel 设为静态,并将其用作维护实例状态的一种方式。