Android Architecture Component ViewModel

Firoz Anwar
5 min readAug 30, 2019

--

Disclaimer:- The information contained in this article are obtained from reliable sources like different websites, blogs and eBooks. The purpose of this blog to have a quick and closer look of this topic. This article is specially designed for experienced professionals, not for beginners. If you are a beginner or experienced developer and want to deep dive you can checkout links provided below in the last of this article.

Problems:-

Rotating a device, configuration changes cause the Activity to be torn down and recreated and data loss.

Another problem is that UI controllers frequently need to make asynchronous calls that may take some time to return. The UI controller needs to manage these calls and ensure the system cleans them up after it’s destroyed to avoid potential memory leaks. This management requires a lot of maintenance, and in the case where the object is re-created for a configuration change, it’s a waste of resources since the object may have to reissue calls it has already made.

Solution:-

At Google I/O 2017, the Android Framework Team introduced a new set of Architecture Components, one of which deals with this exact rotation issue. That is ViewModel. Android architecture components are a collection of libraries that help you design robust, testable, and maintainable apps. Architecture components are part of Android Jetpack that persist data, manage lifecycle, make your app modular, help you avoid memory leaks, and prevent you from having to write boring boilerplate code.

Start with classes for managing your UI component lifecycle and handling data persistence. ViewModel is an Android architecture component. It is used as a data manager in the application. Architecture Components provides ViewModel helper class for the UI controller that is responsible for preparing data for the UI. ViewModel objects are automatically retained during configuration changes so that data they hold is immediately available to the next activity or fragment instance.

ViewModel class is designed to hold and manage UI-related data in a life-cycle conscious way. This allows data to survive configuration changes such as screen rotations. Data gets stored somewhere else, outside of the Activity. This is the purpose of the ViewModel class. ViewModels are very nifty for separating out your UI controller code from the data which fills your UI. A ViewModel provides a way to create and retrieve objects. It typically stores the state of a view’s data and communicates with other components.

Simple Rule: Don’t let your android classes handle everything. Especially not data.

Keep the logic in Activities and Fragments to a minimum.

UML diagram for ViewModel Creation

Closer look and Implementation:-

  • ViewModelProviders.of(<Your UI controller>).get(<Your ViewModel>.class)
  • There are three steps to setting up and using a ViewModel:
  1. Separate out your data from your UI controller by creating a class that extends ViewModel.
  2. Set up communications between your ViewModel and your UI controller.
  3. Use your ViewModel in your UI controller.
  • The ViewModel exists from when you first request a ViewModel (usually in the onCreate the Activity) until the Activity is finished and destroyed. onCreate() may be called several times during the life of an Activity, such as when the app is rotated, but the ViewModel survives throughout.
  • In general, you’ll make a ViewModel class for each screen in your app.
  • ViewModels should not, though, hold a reference to Activities, Fragments, or Contexts. i.e, ViewModels should not contain elements that contain references to UI controllers, such as Views, since this will create an indirect reference to a Context.
  • The first time the ViewModelProviders.of() method is called by Activity, it creates a new ViewModel instance. When this method is called again, which happens whenever onCreate is called, it will return the pre-existing ViewModel associated with the specific Activity. This is what preserves the data.
  • There’s one exception to the “no contexts in ViewModels” rule. Sometimes you might need an Application context (as opposed to an Activity context) for use with things like system services or database.In fact, if you need an Application context, you should extend AndroidViewModel which is simply a ViewModel that includes an Application reference.
  • The ViewModel class does keep track of the associations between ViewModel and UI controller instance behind the scenes, using the UI controller you pass in as the first argument.
  • If the activity is re-created, it receives the same MyViewModel instance that was created by the first activity. When the owner activity is finished, the framework calls the ViewModel objects’s onCleared() method so that it can clean up resources.
  • If the 2 fragments in same activity wants to communicate themselves, they can share the same ViewModel object to easily communicate. Example given below link https://developer.android.com/topic/libraries/architecture/viewmodel
  • ViewModel can replace Loaders like cursorloader to sync the UI with data with the help of ViewModel, LiveData and Room.
  • HolderFragment is a regular Android Headless Fragment. It is the scope where all ViewModels inside the ViewModelStore will live.
  • ViewModel vs onSaveInstanceState() :- ViewModel does not replace onSaveInstanceState(). In onSaveInstanceState() we can store only a small amount of data and data needs to be Parcelable, so it’s not so easy to set and restore values. Use of onSaveInstanceState() to store data necessary for getting data for activity after it’s killed by the system (e.g. id of the current user)
  • Leaking ViewModels:- If the repository is holding a reference to a callback in the ViewModel, the ViewModel will be temporarily leaked. Here is the solution.. With ViewModel.onCleared() you can tell the repository to drop the callback to the ViewModel. (or) In the repository you can use a WeakReference or you can use an Event Bus (both easy to misuse and even considered harmful). (or) Use the LiveData to communicate between the Repository and ViewModel in a similar way to using LiveData between the View and the ViewModel.
  • Dependencies:-

dependencies {

def lifecycle_version = “2.0.0”

// ViewModel and LiveData

implementation “androidx.lifecycle:lifecycle-extensions:$lifecycle_version”

// alternatively — just ViewModel

implementation “androidx.lifecycle:lifecycle-viewmodel:$lifecycle_version”

// alternatively — just LiveData

implementation “androidx.lifecycle:lifecycle-livedata:$lifecycle_version”

Or implementation “android.arch.lifecycle:extensions:1.0.0”

annotationProcessor “android.arch.lifecycle:compiler:1.0.0”

}

Reference Links:-

FAQs:-

--

--

Firoz Anwar

Android Technical Lead at Paytm | IT & Mobile passionate | Writer | Traveler | Learner | Foodie