“This is the first day of my participation in the First Challenge 2022. For details: First Challenge 2022.”


To display the data in Compose, we typically store it in variables and wrap it in mutableStateOf for automatic listening and updates. This step can be written in one of three ways:

val name = mutableStateOf("Bob")               / / 1
val name by mutableStateOf("Bob")              / / 2
val name by remember { mutableStateOf("Bob")}/ / 3
Copy the code

What’s the difference between them? How to choose? This section starts with the most basic question — what does mutableStateOf() actually do?

Compose process

Before delving deeper, it’s worth understanding the full process of Compose from build to display.

Compose consists of three steps: Compose, layout, and draw. The former is unique to Compose and the latter two are similar to traditional Views. By Composition, we create an actual interface based on the code we write. The result of Composition is called Composition.

As an aside, the Compose function, while it looks a lot like creating an object such as Text(Text = name.value), isn’t. TextView().text = name.value ❌ this is not the same as TextView().text = “XXX” in a traditional View.

— So you need to “combine”

If that still doesn’t make sense, you can think of composing as inflat in traditional Views, parsing XML files to create actual objects. It’s just that it used to be possible to create views manually, bypassing XML.

MutableState

The basic use case looks something like this:

val name = mutableStateOf("Bob")
setContent {
    Column {
        Text(text = name.value)
        Button(onClick = { name.value = "2" }) {
            Text(text = "Change Name")}}}Copy the code

Define a Text and Button, click the Button and the displayed Text changes.

Easily see tracking source, mutableStateOf () is to create a ParcelableSnapshotMutableState object and returns it. ParcelableSnapshotMutableState itself is not what something of value, mainly on the implementation of parcelable facilitate the process of communication. The actual data store is its parent SnapshotMutableStateImpl:

internal open class SnapshotMutableStateImpl<T>(
    value: T,
    override val policy: SnapshotMutationPolicy<T>
) : StateObject, SnapshotMutableState<T> {
    override var value: T
        get() = next.readable(this).value
        set(value) = next.withCurrent {
            if(! policy.equivalent(it.value, value)) { next.overwritable(this, it) { this.value = value }
            }
        }

    private var next: StateStateRecord<T> = StateStateRecord(value)

    override val firstStateRecord: StateRecord
        get() = next
    // omit other...
}
Copy the code

Obviously value is the thing that holds the actual data, but it’s still not the final implementation. Exploring further leads to a key class, StateRecord (of which StateStateRecord is a subclass), which you can first understand by looking at the two interfaces that SnapshotMutableStateImpl implements. We now know that mutableStateOf() can return a container object that can be subscribed to and notify subscribers when internal values change, thus implementing UI auto-refresh. The “subscribed” feature is actually provided by the StateObject interface, which is counter-intuitive.

The StateObject itself is a container that stores a string of StateRecord ⬅️ as a linked list, which actually holds the data. Why a string? This is because it not only stores the most recent value of a variable, but also the previous value for “undo” and other functions. Now that is a linked list, it must first node, is StateObject. FirstStateRecord, subsequent node is StateRecord. Next. Pay attention to the next and SnapshotMutableStateImpl. Has nothing to do next, just wish. In fact SnapshotMutableStateImpl a chain table holds all the values in the realized StateObject interface (ah), it is the first node SnapshotMutableStateImpl. Next. SnapshotMutableState

Summary:StateObjectActually stored the data in a linked list,StateRecordAre the nodes of the linked list,SnapshotMutableStateImplIs an implementation.

get

The GET operation calls Readable to retrieve StateRecord and return the actual value of its internal wrapper.

There are three overloaded versions of readable(). The single-parameter version is called each time the value is evaluated, the two-parameter version is called internally, and finally the three-parameter version is called to get the true value.

  • Single-argument: A two-argument convenience function
  • Double parameter: Record this usage (subscription)
  • Three parameters: traverse the StateRecord list to find the latest available StateRecord

“Up to date” is easy to understand, but “effective” goes further

The two-parameter function is at the heart of the automatic UI refresh. Make sure that this action is recorded every time the value is evaluated and updated when the value changes.

set

This function is simpler than calling the three-argument readable() method to find the latest available StateRecord as an argument to execute the lambda expression:

next.withCurrent { it: StateStateRecord<T> ->
    // It is the latest available StateRecord
    if(! policy.equivalent(it.value, value)) { next.overwritable(this, it) { this.value = value }
    }
}
Copy the code

Lambda decides to perform an assignment if the new and old values are not equal. Corresponding to the value, we assume that overWritable implements notification mechanism. Specific tracking to see:

internal inline fun <T : StateRecord, R> T.overwritable(
    state: StateObject,
    candidate: T,
    block: T. () - >R
): R {
    var snapshot: Snapshot = snapshotInitializer
    return sync {
        snapshot = Snapshot.current
        this.overwritableRecord(state, snapshot, candidate).block()
    }.also {
        notifyWrite(snapshot, state)
    }
}

internal fun <T : StateRecord> T.overwritableRecord(
    state: StateObject,
    snapshot: Snapshot,
    candidate: T
): T {
    if (snapshot.readOnly) {
        // If the snapshot is read-only, use the snapshot recordModified to report it.
        snapshot.recordModified(state)
    }
    / / (1) -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
    val id = snapshot.id
    if (candidate.snapshotId == id) return candidate

    / / (2) -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
    val newData = newOverwritableRecord(state, snapshot)
    newData.snapshotId = id
    snapshot.recordModified(state)
    return newData
}

internal fun notifyWrite(snapshot: Snapshot, state: StateObject){ snapshot.writeObserver? .invoke(state) }Copy the code

There is a new concept called Snapshot. As mentioned earlier, the StateRecord is a string of nodes that correspond to the value of Compose in different states. A variable corresponds to a string of StateRecord. Which StateRecord nodes belong to the same state? This is recorded by Snapshot. A StateRecord can correspond to only one Snapshot, and a Snapshot can correspond to multiple Staterecords. Multiple Snapshots are sequential and the latter is built on top of the former. Here’s an example:

As shown in the figure, Record name corresponds to Snapshot1. Snapshot2 is based on S1. Although S2 does not directly correspond to name, “Bob” can be obtained if the value of name is taken in S2, because Snapshot1 of name is valid for Snapshot2.

Return StateRecord if it corresponds directly to Snapshot. Otherwise, create a new StateRecord and return it. Always one way or another, you have to return a corresponding one and you’re done. The details will be studied later.

With this in mind, the first part of OverWritable () becomes clear: save the new value to the StateRecord corresponding to the current Snapshot.

Also block as the name implies is notification! It finds all the places where this variable was read and marks them as invalid. So in the next frame these things will recompose.

Subscribe to? Notice?

Some of you might have found something here. Readable is a subscription and OverWritable is a notification. From the code, however, both invoke() to an Observer, that is, both are “notifications”. WTF?

In fact, the snapshot. ReadObserver? .invoke(state) is both a notification and a subscription. Compose’s subscription mechanism actually has two parts.

The first part

The first part is the subscription to StateObject read and write events in Snapshot. The “notifications” of these two events are the two invoke calls just mentioned. The “subscribe” occurs when Snapshot is created, and the subscribe code is not analyzed in this section.

⚠️ Note: writeObserver will only be notified if a change is made in the compose process, consider the following code:

val name = mutableStateOf("Bob")
setContent {
    Box(Modifier.clickable { name.value = "2" }) {
        Text(name.value) / / 1.
        name.value = "1" / / 2.}}Copy the code

① A read event of name occurred and was recorded. (2) The writeObserver is notified when a write event occurs in the compose process. When we click on Box, we change the name as well, but it’s not in the compose process, so we don’t trigger a write notification.

Although Box is not notified when clicked, the Apply event is subscribed to in the Read event (see below). The UI is still going to refresh.

The second part

The second part is the subscription for each StateObject “Apply” event. This subscription is made when the “read” event occurs. So the snapshot. ReadObserver? .invoke(State) is itself a notification that invokes an observer that “reads” the event. What happens inside the observer is that it subscribes to the “apply” event. So where to notify? I haven’t figured it out yet.

Speaking English: I told you to hand out your annual bonus tomorrow, and you called apple’s flagship store and said to let you know as soon as the MBP arrived. So what I’m doing is both notification and triggering a new subscription.

By “applying,” we mean making the new value globally valid. Can be analogous to the commit of a transaction in an SQL operation.

summary

SnapshotMutableStateImpl stores a list of staterecords. Each node is the value of this variable in a certain state, corresponding to a Snapshot. Snapshot is sequential and stores the values of all variables in a given state.

A “read” event occurs during the get process, and a subscription to the “apply” event is implemented in the listener.

A “write” event occurred during the SET process, but it is only in the compose process that this event actually fires, marking the UI as invalid.

Because Compose has two subscription mechanisms, changes are not made during Compose and the UI is refreshed by the “apply” event.

By the keyword

So now that we’ve solved the first problem, let’s see what happens when we replace theta with by.

We wrapped the value in order to refresh automatically, so that every time we read or write, we had to add a.value attribute. And even mutable values can be declared as val, which is confusing. To solve these problems, kotlin’s native delegate syntax is used.

Val test by test () means that the test variable is read and written by the object on the right. To do this, the right-hand object must have two functions with fixed names and arguments as read/write implementations:

operator fun getValue(thisRef: Any? , property:KProperty< * >): String {
    return "$thisRef, thank you for delegating '${property.name}' to me!"
}

operator fun setValue(thisRef: Any? , property:KProperty<*>, value: String) {
    println("$value has been assigned to '${property.name}' in $thisRef.")}Copy the code

Trace into the source code, you can clearly see the implementation of these two:

inline operator fun <T> State<T>.getValue(thisObj: Any? , property:KProperty< * >): T = value

inline operator fun <T> MutableState<T>.setValue(thisObj: Any? , property:KProperty<*>, value: T) {
    this.value = value
}
Copy the code

In this way, we can assign values to wrapped objects as if we were using native variables, and a mutable variable must be declared as var.

So answer the question: How to choose? A: Just use “by”! Why bother when it’s easy?

The latter is discussed in the next section.