[译]Android架构组件 – 查看ViewModel – 第二部分

原文链接:https://riggaroo.co.za/android-architecture-components-looking-viewmodels-part-2/

如果你回忆上一篇文章,下图指出了我们将如何组织我们的“日期倒计时”应用。

[译]Android架构组件 – 查看ViewModel – 第二部分_第1张图片

本文我们将创建上图显示的EventListViewModelAddEventViewModel。可在这篇文章找到所有源码。在开始创建ViewModels之前,我们需要首先看看什么是ViewModel

什么是ViewModel?

ViewModel既不是新的概念,也不是Android的概念。 名ViewModel来自于2005年左右的Microsoft设计的MVVM模式。新的架构组件中,一个新类是ViewModel类。

ViewModel负责为View准备数据。它们将数据暴露给正在监听更改的任何视图。在Android中,使用ViewModel类时应该记住一些具体的事实:

  • ViewModel可以在Activity配置更改中保留其状态。它保存的数据立即可用于下一个Activity实例,而需要在onSaveInstanceState()中保存数据,并手动还原。
  • ViewModel与特定的Activity或Fragment实例无关。
  • ViewModel允许在Fragment之间轻松共享数据(意味着您不再需要通过ctivity来协调动作)。
  • ViewModel将保持在内存中,直到Lifecycle范围永远消失 - Activity调用finish; 在Fragment调用ditached。
  • 因为ViewModel独立于Activity或Fragment实例,它不直接引用其中的任何View或保持上下文的引用。真会导致内存泄漏。
  • 如果ViewModel需要应用的上下文(例如查找系统服务),它可继承AndroidViewModel类,并有一个构造函数来接收Application实例。

创建ViewModel

EventListViewModel

EventListViewModel类用于打开日期倒计时应用打开时展示事件列表。

  1. 创建名为EventListViewModel的类。确保它从架构组件类继承ViewModel。类中,我们打算将之前放在EventListFragment中的代码迁移到ViewModel中。
  2. EventListViewModel中添加LiveData变量。EventRepository变量使用Dagger注入。EventListViewModel现在长这样:

    public class EventListViewModel extends ViewModel implements CountdownComponent.Injectable {
    
        private LiveData> events = new MutableLiveData<>();
    
        @Inject
        EventRepository eventRepository;
    
        @Override
        public void inject(CountdownComponent countdownComponent) {
            countdownComponent.inject(this);
            events = eventRepository.getEvents();
        }
    
        public LiveData> getEvents() {
            return events;
        }
    }
  3. 现在在EventListFragment中,我们打算使用ViewModel替换事件加载代码:

    public class EventListFragment extends LifecycleFragment {
    
        private EventListViewModel eventListViewModel;
    
        @Nullable
        @Override
        public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
            //.. inflate view etc
            eventListViewModel = ViewModelProviders.of(this, new CountdownFactory(countdownApplication).get(EventListViewModel.class);
    
            eventListViewModel.getEvents().observe(this, events -> {
                adapter.setItems(events);
            });
            return v;
        }
    }

    包含ViewModelProvider.of(..)的行将创建一个新的EventListViewModel类(如果不存在),或者它将获取存在于定义的作用域(在这种情况下是EventListFragment)的类。这是神奇的地方。当设备旋转并且Fragment被重新创建时,将在此处返回的ViewModel是之前使用的。这允许我们保留屏幕的状态,而无需手动将信息保存到Bundle并自己恢复。
    事件数据现在从EventListViewModel获取。如前所述,通过将其作为第一个参数,LiveData observable将自动为您管理。这意味着当Fragment不再使用时,Fragment将会处理可观察物。如果Fragment未启动或恢复,则不可调用observable回调。

  4. 创建一个Dagger组件CountdownComponent。我们将使用它来注入ViewModel的依赖。

    @Singleton
    @Component(modules = {CountdownModule.class})
    public interface CountdownComponent {
    
        void inject(EventListViewModel eventListViewModel);
    
        void inject(AddEventViewModel addEventViewModel);
    
        interface Injectable {
            void inject(CountdownComponent countdownComponent);
        }
    }
  5. 创建自定义ViewModelProvider.NewInstanceFactory用于实例化ViewModel。当ViewModelProvider需要创建新的ViewModel实例,它将调用工厂类的create方法。

    public class CountdownFactory extends ViewModelProvider.NewInstanceFactory {
    
        private CountdownApplication application;
    
        public CountdownFactory(CountdownApplication application) {
            this.application = application;
        }
    
        @Override
        public  T create(Class modelClass) {
            T t = super.create(modelClass);
            if (t instanceof CountdownComponent.Injectable) {
                ((CountdownComponent.Injectable) t).inject(application.getCountDownComponent());
            }
            return t;
        }
    }
  6. 删除Event,可以在ViewModel中添加一个方法。该方法委托EventRepository删除事件。本例中,使用Rxjava来代理后台线程。

    public class EventListViewModel extends ViewModel implements CountdownComponent.Injectable {
    //.. 
    public void deleteEvent(Event event) {
        eventRepository.deleteEvent(event)
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new CompletableObserver() {
                    @Override
                    public void onSubscribe(Disposable d) {
                    }
    
                    @Override
                    public void onComplete() {
                        Timber.d("onComplete - deleted event");
                    }
    
                    @Override
                    public void onError(Throwable e) {
                        Timber.e("OnError - deleted event: ", e);
                    }
                });
        }
    }

在Fragment中,我们使用委托给ViewModel的删除方法:

View.OnClickListener deleteClickListener = v -> {
        Event event = (Event) v.getTag();
        eventListViewModel.deleteEvent(event);
};

对于AddEventViewModel的实现,查看完整代码示例。

总结

新的Android架构组件解决了我们以前无法处理的常见情况。现在通过使用ViewModelViewModelProvider来处理屏幕旋转非常简单。

参考

  • ViewModel文档
  • Android架构组件文档
  • 日期倒计时应用-完整代码
  • Dagger2 测试实例

你可能感兴趣的:(Android)