Introduction and Use

LiveData is officially defined as: an observable data store class; Has the following advantages:

  • The observer mode ensures that the interface conforms to the data state without updating the interface when the data changes.
  • With life cycle awareness, there is no need to manually process the life cycle, nor will the Activity stop and cause a crash;
  • The observer is bound to Lifecycle and will be cleaned up automatically without memory leaks;
  • Data is always in the latest state, the life cycle becomes inactive, and it displays the latest data when it becomes active again (the latest data is displayed when the Activity life cycle changes and it returns to the active state)

Basic usage:

Create:

class LiveDataViewModel : ViewModel() {
    val currentCount: MutableLiveData<Int> by lazy {
        MutableLiveData(0)}}Copy the code

Use:

class LiveDataActivity : AppCompatActivity() {
    val viewModel: LiveDataViewModel by viewModels()
    override fun onCreate(savedInstanceState: Bundle?). {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_live_data)
        val textView = findViewById<TextView>(R.id.tv1)
        // Add a listener to observe the LiveData object
        viewModel.currentCount.observe(this, Observer<Int> {
            textView.text = "$it"
        })
        findViewById<Button>(R.id.button).setOnClickListener {
        	// Update LiveData object dataviewModel.currentCount.value = viewModel.currentCount.value? .plus(1)}}}Copy the code

As this article focuses on LiveData, design into LifeCycle will not go into the source code in detail.

Source code analysis

Note the advantages of LiveData that we introduced at the beginning: observer mode, lifecycle binding and auto-destruct cleanup, and the fact that the data is always up to date. Let us explore these three points in detail here

Observer creation and lifecycle binding

Observe (LifecycleOwner owner, Observer
observer) to see how LiveData adds an observer:

public void observe(@NonNull LifecycleOwner owner, @NonNull Observer<? super T> observer) {
	/ / step one
    assertMainThread("observe");
    if (owner.getLifecycle().getCurrentState() == DESTROYED) {
        return;
    }
    / / in step 2
    LifecycleBoundObserver wrapper = new LifecycleBoundObserver(owner, observer);
    ObserverWrapper existing = mObservers.putIfAbsent(observer, wrapper);
    / / step 3
    if(existing ! =null && !existing.isAttachedTo(owner)) {
        throw new IllegalArgumentException("Cannot add the same observer"
                + " with different lifecycles");
    }
    / / step 4
    if(existing ! =null) {
        return;
    }
    owner.getLifecycle().addObserver(wrapper);
}
Copy the code

There are three steps in the code:

  • Step 1: First determine if you are currently in the live thread (i.e., the UI thread). If not, an exception is thrown directly; Then check the Activity lifecycle if it is in the DeStory phase. So just go back and do nothing; (LifecycleOwner has an Android life cycle and LiveData can use these events to handle Activty or Fragement life cycles)
  • Step 2: CreateLifecycleBoundObserver, the use ofobserver It will be added as the Key valuewrapper Added to themObserversIn the.
  • Step 3: Determine whether the current observer is doubly bound. While avoiding the same observer (Observer) Use two different onesLifecycleOwner
  • Step 4: If the observer is not double-bound, bind it withLifecycleAssociation.

First look at the Observer.

public interface Observer<T> {
    /**
     * Called when the data is changed.
     * @param t  The new data
     */
    void onChanged(T t);
}
Copy the code

The code is very simple, just an interface. An onChanged method is used to notify that data has changed.

Next, LifecycleBoundObserver: it inherits from ObserverWrapper and implements the LifecycleEventObserver interface.

class LifecycleBoundObserver extends ObserverWrapper implements LifecycleEventObserver {
    @NonNull
    final LifecycleOwner mOwner;

    LifecycleBoundObserver(@NonNull LifecycleOwner owner, Observer<? super T> observer) {
        super(observer);
        mOwner = owner;
    }

    @Override
    boolean shouldBeActive(a) {
        return mOwner.getLifecycle().getCurrentState().isAtLeast(STARTED);
    }

    @Override
    public void onStateChanged(@NonNull LifecycleOwner source,
            @NonNull Lifecycle.Event event) {
        Lifecycle.State currentState = mOwner.getLifecycle().getCurrentState();
        if (currentState == DESTROYED) {
            removeObserver(mObserver);
            return;
        }
        Lifecycle.State prevState = null;
        while (prevState != currentState) {
            prevState = currentState;
            activeStateChanged(shouldBeActive());
            currentState = mOwner.getLifecycle().getCurrentState();
        }
    }

    @Override
    boolean isAttachedTo(LifecycleOwner owner) {
        return mOwner == owner;
    }

    @Override
    void detachObserver(a) {
        mOwner.getLifecycle().removeObserver(this); }}Copy the code

We can think of it as a wrapper class for an Observer and LifeCycle, which holds the power of the Observer Observer and LifecycleOwner. I won’t go into the details of its methods (more on them later).

Moving on to step 2: The wrapper class created is put into mObservers, which is a member variable of Type SafeIterableMap of LiveData — it is obviously used to manage all observers (by the wrapper class that holds them). Without going into the details of SafeIterableMap, you can roughly think of it as a HashMap (it’s not, but that’s good enough for you to understand LiveData). Let’s talk about the putIfAbsent method:

public V putIfAbsent(@NonNull K key, @NonNull V v) { Entry<K, V> entry = get(key); if (entry ! = null) { return entry.mValue; } put(key, v); return null; }Copy the code

If the map does not contain the current key, add it to the map and return NULL. Otherwise, the current key value is returned. Also, since the ObserverWrapper is created (that is, LifecycleBoundObserver above), it will hold the LifecycleOwner. Therefore, with the isAttachedTo method:

boolean isAttachedTo(LifecycleOwner owner) {
    return false;
}
Copy the code

Step 3: Avoid using two different Lifecycleowners for the same Observer

Here’s the onStateChanged method. It is a method defined in the LifecycleEventObserver interface. With this interface, you can perform step 4: owner.getLifecycle().addobServer (wrapper); Implement listening for the lifecycle. LifecycleBoundObserver methods are called whenever the Activity/Fragment life cycle changes. Unlike a ViewModel, the lifecycle of a ViewModel is entirely managed by its host Activity/Fragment. The listening lifecycle approach is used directly here. Android ViewModel (ViewModel)

At this point, our observer is ready and associated with the lifecycle of the host Activity/Fragment. Next, it’s up to the observer to respond.

The flow of events as data changes

In the opening use, we use setValue of LiveData to realize data update:

protected void setValue(T value) {
    assertMainThread("setValue");
    mVersion++;
    mData = value;
    dispatchingValue(null);
}
Copy the code

The method is very simple, first determine whether the main thread. Then add mVersion. MVersion is a member variable that records how many times the data has changed. The initial value is -1 and +1 each time the data is changed. MData is the data held and managed by LiveData. After updating the data, it is distributed:

void dispatchingValue(@Nullable ObserverWrapper initiator) { ...... do { mDispatchInvalidated = false; if (initiator ! = null) { ...... } else { for (Iterator<Map.Entry<Observer<? super T>, ObserverWrapper>> iterator = mObservers.iteratorWithAdditions(); iterator.hasNext(); ) { considerNotify(iterator.next().getValue()); if (mDispatchInvalidated) { break; } } } } while (mDispatchInvalidated); mDispatchingValue = false; }Copy the code

Omit the code of the assignment verification operation. Here, we only see the for loop in the case that the initiator is Null. The code is simple, just loop through the mObservers — all of our observer wrap objects — and call let notify:

private void considerNotify(ObserverWrapper observer) {
    if(! observer.mActive) {return;
    }
    if(! observer.shouldBeActive()) { observer.activeStateChanged(false);
        return;
    }
    if (observer.mLastVersion >= mVersion) {
        return;
    }
    observer.mLastVersion = mVersion;
    observer.mObserver.onChanged((T) mData);
}
Copy the code

The method first determines whether the observer is active and ensures that the observer is notified only when the observer is active. Then determine the lifecycle of the observer and ensure that it is not notified of data changes until it is active. Finally, the viewer is compared to the version of the data in LiveData and decides whether to notify the viewer.

So that’s what happens when the data changes. The whole is a simple observer model.

How do I keep my data up to date

I don’t know if you’ve written code like this:

viewModel.currentCount.observe(this, Observer<Int> {
    Toast.makeText(this."$it",Toast.LENGTH_SHORT).show()
})
Copy the code

If you use this code in your Activity, you’ll notice that Toast always pops up when you rotate the screen. What’s going on here? Many articles will talk about this phenomenon as a weakness of LiveData, saying that “data stickiness”, “flooding” and other new terms. He even “criticized” it as a crime against LiveData.

In fact, this is one of the features of LiveData. The official definition is: ** Data is always up to date: if a life cycle becomes inactive, it receives the latest data when it becomes active again. For example, an Activity that was once in the background receives the latest data as soon as it returns to the foreground. ** This is one of the features of LiveData.

So how does that work? Remember LifecycleBoundObserver above? We said that onStateChanged is called when the host’s life cycle changes:

@Override
public void onStateChanged(@NonNull LifecycleOwner source,
        @NonNull Lifecycle.Event event) {
    Lifecycle.State currentState = mOwner.getLifecycle().getCurrentState();
    if (currentState == DESTROYED) {
        removeObserver(mObserver);
        return;
    }
    Lifecycle.State prevState = null;
    while (prevState != currentState) {
        prevState = currentState;
        activeStateChanged(shouldBeActive());
        currentState = mOwner.getLifecycle().getCurrentState();
    }
}
Copy the code

The key is the while loop, which first determines the current life cycle and avoids doing things when the life cycle is unchanged. Then call activeStateChanged:

void activeStateChanged(boolean newActive) { if (newActive == mActive) { return; } mActive = newActive; changeActiveCounter(mActive ? 1:1); if (mActive) { dispatchingValue(this); }}Copy the code

You can see here that you are familiar with the method dispatchingValue, which is also called when data changes above. The only difference is that there is a parameter passed: ObserverWrapper, which is our ObserverWrapper class. So take a look at how dispatchingValue is handled:

void dispatchingValue(@Nullable ObserverWrapper initiator) { if (mDispatchingValue) { mDispatchInvalidated = true; return; } mDispatchingValue = true; do { mDispatchInvalidated = false; if (initiator ! = null) { considerNotify(initiator); initiator = null; } else { for (Iterator<Map.Entry<Observer<? super T>, ObserverWrapper>> iterator = mObservers.iteratorWithAdditions(); iterator.hasNext(); ) { considerNotify(iterator.next().getValue()); if (mDispatchInvalidated) { break; } } } } while (mDispatchInvalidated); mDispatchingValue = false; }Copy the code

When data changes, dispatchingValue traverses all observers to inform them of data updates. When the life cycle changes, dispatchingValue notifies the observer separately according to the parameter’s ObserverWrapper. So how can all observers be notified? LifeCycle is actually handed over to LifeCycle. As mentioned earlier in the observer creation process, every time a LifecycleBoundObserver is created, a LifeCycle listener is added to LifeCycle via owner.getLifecycle().addobServer (Wrapper). LifeCycle notifies all LiveData observers of changes in the LifeCycle by traversing these listeners.

Some auxiliary methods

Observe (@nonnull LifecycleOwner owner, @nonnull Observer
observer) method, LiveData also provides observeForever(@nonnull observer
observer) method. The main difference is that observeForever gets notified when the LiveData wrapper changes, regardless of the state of the page.

public void observeForever(@NonNull Observer<? super T> observer) { assertMainThread("observeForever"); AlwaysActiveObserver wrapper = new AlwaysActiveObserver(observer); ObserverWrapper existing = mObservers.putIfAbsent(observer, wrapper); if (existing instanceof LiveData.LifecycleBoundObserver) { throw new IllegalArgumentException("Cannot add the same observer" + " with different lifecycles"); } if (existing ! = null) { return; } wrapper.activeStateChanged(true); }Copy the code

ObserveForever obviously doesn’t accept the LifecycleOwner parameter, so it can’t listen for the lifecycle. Its ObserverWrapper class looks like this:

private class AlwaysActiveObserver extends ObserverWrapper { AlwaysActiveObserver(Observer<? super T> observer) { super(observer); } @Override boolean shouldBeActive() { return true; }}Copy the code

Without implementing LifecycleEvening Server, there is naturally no way to listen for lifecycle changes and notify observers of updates. Also because you cannot listen to the lifecycle, you must use removeObserver to remove the listener.

LiveData provides two methods to remove listeners: removeObserver(@nonnull final Observer
Observer) and removeObservers(@nonnull final LifecycleOwner owner). The former removes the specified observer, while the latter removes all observers that are bound to the current page life cycle.

LiveData also provides a way to update data from non-UI threads: postValue:

private final Runnable mPostValueRunnable = new Runnable() {
    @SuppressWarnings("unchecked")
    @Override
    public void run() {
        Object newValue;
        synchronized (mDataLock) {
            newValue = mPendingData;
            mPendingData = NOT_SET;
        }
        setValue((T) newValue);
    }
};

protected void postValue(T value) {
    boolean postTask;
    synchronized (mDataLock) {
        postTask = mPendingData == NOT_SET;
        mPendingData = value;
    }
    if (!postTask) {
        return;
    }
    ArchTaskExecutor.getInstance().postToMainThread(mPostValueRunnable);
}
Copy the code

It’s essentially a Handler, and you use Handler to switch to the main thread and then use setValue to update the data.

Summary and reflection

There are many articles on the web about ViewModel+LiveData replacing RxJava. Here is my understanding of LiveData and RxJava (RxAndroid). Start with the argument that the two duties serve different purposes, and strictly speaking, you shouldn’t be talking about who replaces who (more on why strict should be added later in this article).

Personally, Rx is designed to make it as easy to handle asynchronous operations as it is to handle synchronous operations.

However, Google officially gave LiveData a clear definition:

LiveData is an observable data holder class

LiveData is an observable data store class!!

They both have their own uses and should not be compared side by side. Rx is mainly used to help us sort out logic and simplify business code. LiveData stores the data for us.

RxJava is called Reactive Extensions for the JVM. It can be understood as a responsive programming extension to Java. It’s different from generic libraries like Retrofit, OkHttp, etc. Instead of giving us examples of how to solve specific problems, Rx gives us the logic to make asynchrony as simple as synchronization. Its observer, Observable, you can also think of it as a CallBack. The difference is that upstream data producers simply send data via an Observable, either asynchronously or synchronously or several times. Downstream observers should process and not care about how the data was generated. And all the callbacks are on the same line, kind of like a river, or flow.

Let’s go back to how we use it in Android: most of it is to make a network request and switch threads. You don’t even need a few operators. If so, then of course it can be replaced. But is it its fault? No, look at what Rx was designed for, and why are you using it? What you see as its advantages, can it be replaced with sensible code design? Ask yourself, have you considered other options before using this library? How much do you need to use it and how much do you want to follow suit? If you are using it to make a network request, then of course it can be replaced. Because coroutines plus LiveData can be done better and easier to get started.

So in the selection of technology, we must fit the business and reality, do not blindly cite.

Finally, back to LiveData. Now there is a clamour of criticism and talk of replacing LiveData on the Internet. The criticism is based on the following points:

  1. “Data stickiness” : this is the case above where the vertical and horizontal screen toggles to show Toast. The word “data stickiness” is probably invented by netizens, but this is not the fault of LiveData! That’s one of its features! LiveData is an observable data store class that ensures that data is always up to date: if the life cycle becomes inactive, it receives the latest data when it becomes active again. For example, an Activity that was once in the background receives the latest data as soon as it returns to the foreground. That’s one of its great advantages!

    As a data store itself, you should not use it to hold one-time data or events, let alone as an EventBus. It has its own responsibilities and division of labor.

  2. Data loss caused by postValue. Thread scheduling is not good: LiveData should only be used to store data related to UI state, the UI should always show the latest state, and LiveData should always notify the observer of the latest data. You can’t expect it to do anything else. If you want to do a lot of scheduling, work with Rx or Kotlin’s Flow.

  3. The operator is not powerful enough. Note: LiveData’s job is not to process data, but to store it.

In short, before looking at a thing or technology. It must not be removed from its essence, nor imposed on it attributes or functions before it has been analyzed. As for RxJava being phased out, LiveData has been replaced by Flow. I’m on the other side. First of all, the three have different emphasis and design intentions. Second, Rx and Flow are both language extensions, while LiveData is a data management solution at the Android level — how can there be overlap and substitution?

Personally, I think it’s a symbiotic relationship, each doing what he’s good at. Combined to create a 1+1>>2 effect.

As for the other drawbacks, I would say that setValue updates directly without checking the replay is more of a disadvantage than the above ones. Of course, we can solve this problem by referring to rewrite method:

class CustomLiveData : LiveData<String>() { override fun setValue(value: String?) { if (! value.equals(getValue())) { super.setValue(value) } } }Copy the code

All of the above is my humble opinion, and I hope to correct the fallacy. Also welcome everybody big guy message discussion, express different opinion actively. We learn from each other and make progress together.

How to use viewModels in custom Views

Android ViewModel source code analysis