The ViewModel profile

In the last article we looked at the use and rationale of Jetpack Lifecycle. Today we will look at the use and rationale of Jetpack ViewModel components.

The ViewModel class is designed to manage and store UI data in a life-cycle aware manner, such as when the screen rotates, App permissions are dynamically changed, and the system language changes (all of which are Configuration changes). When an Activity is rebuilt, the ViewModel can retain data from the interface. Before ViewModel, the onSaveInstanceState() method is used to store data in the Bundle, and the onCreate() method is used to determine whether the Bundle is empty or not. If not, the stored data is retrieved from the Bundle. If the data to be saved is of complex type, Serializable or Parcelable interfaces need to be implemented. Therefore, saving and restoring data in this way is cumbersome and not suitable for storing large amounts of data. With the ViewModel, it’s a lot easier. You don’t have to manually save data in a particular method, the action system does that automatically, and the ViewModel can hold any object, Complex types do not need to implement Serializable or Parcelable interfaces.

In order to better understand the ViewModel, we will not introduce other concepts such as LiveData.

Note: The ViewModel used in this article is the latest version 2.2.0.

The introduction of the ViewModel

First we’ll introduce viewModel-related libraries, if your project is Kotlin:

Dependencies {def lifecycle_version = "implementation" "androidx.lifecycle:lifecycle-viewmodel-ktx:$lifecycle_version" // Saved state module for ViewModel implementation "androidx.lifecycle:lifecycle-viewmodel-savedstate:$lifecycle_version" }Copy the code

If your project is Java:

Dependencies {def lifecycle_version = "implementation" "androidx.lifecycle:lifecycle-viewmodel:$lifecycle_version" // Saved state module for ViewModel implementation "androidx.lifecycle:lifecycle-viewmodel-savedstate:$lifecycle_version" }Copy the code

The ViewModel case

The Demo is also very simple. Click the button, and the interface keeps adding numbers. Then when the screen rotates the Activity to rebuild, the interface still retains the data displayed in the previous Activity instance, as shown below:

ViewModelDemoActivity. Kt:

class ViewModelDemoActivity : androidx.appcompat.app.AppCompatActivity() {

    private lateinit var myViewModel: MyViewModel

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_viewmodel_demo_layout)

        myViewModel = ViewModelProvider(this).get(MyViewModel::class.java)

        text_number.text = "${myViewModel.number}"

        btn_plus.setOnClickListener {
            text_number.text = "${++myViewModel.number}"
        }
    }
}
Copy the code

MyViewModel. Kt:

class MyViewModel : ViewModel() {
    var number: Int = 0
}
Copy the code

The code above is very simple, first let MyViewModel inherit from ViewModel, put the data in MyViewModel, The MyViewModel object is then retrieved in the Activity using the viewModelProvider.get () method. The retrieved ViewModel object is the same as the ViewModel object before the Activity was rebuilt. Therefore, the previous data can be restored after the interface is rotated. Let’s look at how ViewModel works, and why viewModelProvider.get () gets the same ViewModel instance when configuration changes.

ViewModel principle analysis

The most recent key line in the Demo above is:

ViewModelProvider(this).get(MyViewModel::class.java)
Copy the code

Let’s look at the ViewModelProvider constructor first:

public class ViewModelProvider { public ViewModelProvider(@NonNull ViewModelStoreOwner owner) { this(owner.getViewModelStore(), owner instanceof HasDefaultViewModelProviderFactory ? ((HasDefaultViewModelProviderFactory) owner).getDefaultViewModelProviderFactory() : NewInstanceFactory.getInstance()); } public ViewModelProvider(@NonNull ViewModelStore store, @NonNull Factory factory) { mFactory = factory; mViewModelStore = store; }}Copy the code

We pass this to the constructor of ViewModelProvider, which indicates that AppCompatActivity must implement the ViewModelStoreOwner interface, and the inheritance relationship is as follows:

androidx.appcompat.app.AppCompatActivity
    -> androidx.fragment.app.FragmentActivity
        -> androidx.activity.ComponentActivity
            -> androidx.lifecycle.ViewModelStoreOwner
Copy the code

So we’ll look at ComponentActivity. GetViewModelStore:

public class ComponentActivity extends androidx.core.app.ComponentActivity implements LifecycleOwner, ViewModelStoreOwner, HasDefaultViewModelProviderFactory, SavedStateRegistryOwner, OnBackPressedDispatcherOwner {/ / omit other code... @NonNull @Override public ViewModelStore getViewModelStore() { if (getApplication() == null) { throw new IllegalStateException("Your activity is not yet attached to the " + "Application instance. You can't request ViewModel before onCreate call."); } if (mViewModelStore == null) { NonConfigurationInstances nc = (NonConfigurationInstances) getLastNonConfigurationInstance(); if (nc ! = null) { // Restore the ViewModelStore from NonConfigurationInstances mViewModelStore = nc.viewModelStore; } if (mViewModelStore == null) { mViewModelStore = new ViewModelStore(); } } return mViewModelStore; }}Copy the code

The key code for getLastNonConfigurationInstance method, if the returned NonConfigurationInstances object is not null, then obtain ViewModelStore and then return.

So we’ll look at getLastNonConfigurationInstance method:

public class ComponentActivity extends androidx.core.app.ComponentActivity implements
        LifecycleOwner,
        ViewModelStoreOwner,
        HasDefaultViewModelProviderFactory,
        SavedStateRegistryOwner,
        OnBackPressedDispatcherOwner {
        
    /**
     * Retrieve the non-configuration instance data that was previously
     * returned by {@link #onRetainNonConfigurationInstance()}.  This will
     * be available from the initial {@link #onCreate} and
     * {@link #onStart} calls to the new instance, allowing you to extract
     * any useful dynamic state from the previous instance.
     *
     * <p>Note that the data you retrieve here should <em>only</em> be used
     * as an optimization for handling configuration changes.  You should always
     * be able to handle getting a null pointer back, and an activity must
     * still be able to restore itself to its previous state (through the
     * normal {@link #onSaveInstanceState(Bundle)} mechanism) even if this
     * function returns null.
     *
     * <p><strong>Note:</strong> For most cases you should use the {@link Fragment} API
     * {@link Fragment#setRetainInstance(boolean)} instead; this is also
     * available on older platforms through the Android support libraries.
     *
     * @return the object previously returned by {@link #onRetainNonConfigurationInstance()}
     */
    @Nullable
    public Object getLastNonConfigurationInstance() {
        return mLastNonConfigurationInstances != null
                ? mLastNonConfigurationInstances.activity : null;
    }
}
Copy the code

Can find NonConfigurationInstances is mLastNonConfigurationInstances. The activity, Through getLastNonConfigurationInstance () method of annotation as we know, the method used to get onRetainNonConfigurationInstance () method returns the object of “non – configuration”.

So we’ll look at onRetainNonConfigurationInstance () method:

public class ComponentActivity extends androidx.core.app.ComponentActivity implements LifecycleOwner, ViewModelStoreOwner, HasDefaultViewModelProviderFactory, SavedStateRegistryOwner, OnBackPressedDispatcherOwner { /** * Retain all appropriate non-config state. You can NOT * override this yourself! Use a {@link androidx.lifecycle.ViewModel} if you want to * retain your own non config state. */ @Override @Nullable public final Object onRetainNonConfigurationInstance() { Object custom = onRetainCustomNonConfigurationInstance(); ViewModelStore viewModelStore = mViewModelStore; GetViewModelStore = mViewModelStore = mViewModelStore = mViewModelStore; If (viewModelStore == null) {// No one called getViewModelStore(), so see if there was an existing // ViewModelStore from our last NonConfigurationInstance NonConfigurationInstances nc = (NonConfigurationInstances) getLastNonConfigurationInstance(); if (nc ! = null) { viewModelStore = nc.viewModelStore; } } if (viewModelStore == null && custom == null) { return null; } / / encapsulation NonConfigurationInstances NonConfigurationInstances nci = new NonConfigurationInstances (); nci.custom = custom; nci.viewModelStore = viewModelStore; return nci; }}Copy the code

OnRetainNonConfigurationInstance method will be called before interface reconstruction, this is the last time to save viewModelStore examples. Then at the time of interface reconstruction calls onCreate getLastNonConfigurationInstance way before getting retain NonConfigurationInstances object. That we are back to getLastNonConfigurationInstance method:

public class ComponentActivity extends androidx.core.app.ComponentActivity implements
        LifecycleOwner,
        ViewModelStoreOwner,
        HasDefaultViewModelProviderFactory,
        SavedStateRegistryOwner,
        OnBackPressedDispatcherOwner {
   
    @Nullable
    public Object getLastNonConfigurationInstance() {
        return mLastNonConfigurationInstances != null
                ? mLastNonConfigurationInstances.activity : null;
    }
}
Copy the code

Find getLastNonConfigurationInstance method of operation is ComponentActivity mLastNonConfigurationInstances member variables, When then ComponentActivity of mLastNonConfigurationInstances attribute assignment before?

public class Activity extends ContextThemeWrapper implements LayoutInflater.Factory2, Window.Callback, KeyEvent.Callback, OnCreateContextMenuListener, ComponentCallbacks2, Window.OnWindowDismissedCallback, WindowControllerCallback, AutofillManager.AutofillClient, ContentCaptureManager.ContentCaptureClient { @UnsupportedAppUsage final void attach(Context context, ActivityThread aThread, Instrumentation instr, IBinder token, int ident, Application application, Intent intent, ActivityInfo info, CharSequence title, Activity parent, String id, NonConfigurationInstances lastNonConfigurationInstances, Configuration config, String referrer, IVoiceInteractor voiceInteractor, Window window, ActivityConfigCallback activityConfigCallback, IBinder assistToken) {// omit other code... mLastNonConfigurationInstances = lastNonConfigurationInstances; }Copy the code

It was assigned in the Activity. Attach method, which is called when the Activity rebuilds, Last saved lastNonConfigurationInstances object will be assigned to mLastNonConfigurationInstances member variables. So who called Attach? In ActivityThread. PerformLaunchActivity method of trigger the attach:

public final class ActivityThread extends ClientTransactionHandler { private Activity PerformLaunchActivity (ActivityClientRecord r, Intent customIntent) {// activity.attach(appContext, this, getInstrumentation(), r.token, r.ident, app, r.intent, r.activityInfo, title, r.parent, r.embeddedID, r.lastNonConfigurationInstances, config, r.referrer, r.voiceInteractor, window, r.configCallback, r.assistToken); }}Copy the code

Ok, now we know when the mLastNonConfigurationInstances attribute assignment. When the calls onRetainNonConfigurationInstance method?

According to the comments of getLastNonConfigurationInstance method source code as we know, the method to obtain the onRetainNonConfigurationInstance () method returns the object, So who would trigger onRetainNonConfigurationInstance method call? See ActivityThread source that in ActivityThread. PerformDestroyActivity method call:

Public Final Class ActivityThread extends ClientTransactionHandler {// ActivityClientRecord performDestroyActivity(IBinder token, boolean finishing, int configChanges, Boolean getNonConfigInstance, String reason) {// omit other code... ActivityClientRecord r = mActivities.get(token); if (getNonConfigInstance) { try { r.lastNonConfigurationInstances = r.activity.retainNonConfigurationInstances(); } catch (Exception e) { if (! mInstrumentation.onException(r.activity, e)) { throw new RuntimeException( "Unable to retain activity " + r.intent.getComponent().toShortString() + ": " + e.toString(), e); } } } return r; }}Copy the code

So at the time of interface destruction will call retainNonConfigurationInstances used to hold data.

summary

So far, we will be the whole source in series, so the basic principle of ViewModel is: inconfiguration changesWhen in the interfacedestroy(ActivityThread.performDestroyActivity) byonRetainNonConfigurationInstanceMethods NonConfigurationInstances object (containing ViewModel) saved, the reconstruction in the ActivityattachMethod (ActivityThread.performLaunchActivity) will be the last saved NonConfigurationInstances object is assigned to the Activity of mLastNonConfigurationInstances member variables, and then passed in onCreategetLastNonConfigurationInstanceMethod to get the last saved NonConfigurationInstances object.