The brand new sturdy skipping mode for controlling class stability in Jetpack Compose modifications learn how to optimize recompositions in your app. On this weblog submit, we’ll cowl what circumstances it solves for you and what must be manually managed. We’ll additionally cowl frequent questions you’ve had, similar to whether or not remembering lambda features remains to be wanted, if kotlinx immutable collections are wanted, and even learn how to stabilize all of your area mannequin courses. When you’re undecided what stability is, see our documentation to be taught the ideas.
There are a number of the reason why the Compose compiler may deal with a category as unstable:
- It’s a mutable class. For instance, it incorporates a mutable property (not backed by snapshot state).
- It’s a category outlined in a Gradle module that doesn’t use Compose (doesn’t have a dependency on the Compose compiler).
- It’s a category that incorporates an unstable property (instability nesting).
Let’s contemplate the next class:
knowledge class Subscription(          // class is unstable
val id: Int,                  // steady
val planName: String,         // steady
val renewalOn: LocalDate      // unstable
)
The id and planName properties are steady, as a result of they’re of a primitive kind, which is immutable. Nevertheless, the renewalOn property is unstable, as a result of java.time.LocalDate comes from the Java normal library, which doesn’t have a dependency on the Compose compiler. Due to that, the entire Subscription class is unstable.
Contemplate the next instance with a state property utilizing the Subscription class, which is handed to a SubscriptionComposable:
// create in a state holder (for instance, ViewModel)
var state by mutableStateOf(Subscription(
id = 1,
planName = "30 days",
renewalOn = LocalDate.now().plusDays(30)
))@Composable
enjoyable SubscriptionComposable(enter: Subscription) {
// all the time recomposed regardless if enter modified or not
}
Traditionally, a composable with the enter parameter of this unstable class wouldn’t be decided as skippable, and it might all the time be recomposed regardless if the inputs modified or not.
Jetpack Compose compiler 1.5.4 and better comes with an choice to allow sturdy skipping mode, which all the time generates the skipping logic whatever the stability of the enter parameters. This mode permits composables with unstable courses to be skipped. You may learn extra about sturdy skipping mode and learn how to allow it in our documentation or within the weblog submit by Ben Trengrove.
Sturdy skipping mode has two methods of figuring out if the enter parameter modified from the earlier composition:
- If the category is steady, it makes use of the structural equality (.equals()).
- If the category is unstable, it makes use of the referential equality (===).
After you allow sturdy skipping mode in your venture, composables that use the unstable Subscription class gained’t recompose if the occasion is identical as within the earlier composition.
So let’s say you could have the SubscriptionComposable utilized in a unique composable Display that takes a parameter inputText. If that inputText parameter modifications and the subscription parameter doesn’t, the SubscriptionComposable doesn’t recompose and is skipped:
@Composable
enjoyable Display(inputText: String, subscription: Subscription) {
Textual content(inputText)// It is skipped when subscription parameter did not change
SubscriptionComposable(subscription)
}
However let’s say you could have a perform renewSubscription that updates the state variable with the present day to maintain monitor of newest day when a change occurred:
enjoyable renewSubscription() {
state = state.copy(renewalOn = LocalDate.now().plusDays(30))
}
The copy perform creates a new occasion of the category with the identical structural properties (if it happens throughout the identical day), which signifies that the SubscriptionComposable would recompose once more, as a result of sturdy skipping mode compares unstable courses with referential equality (===) and replica is creating a brand new occasion of our subscription. Regardless that the date is identical, as a result of referential equality is getting used, the Subscription composable remains to be recomposed.
If you wish to forestall the SubscriptionComposable from recomposing when the structural knowledge doesn’t change (equals() returns the identical consequence), it’s essential manually mark the Subscription class as steady.
On this case, it’s a easy repair by annotating the category with @Immutable, as a result of the category represented right here can’t be mutated:
+@Immutable           
-data class Subscription(       // unstable
+knowledge class Subscription(       // steady
val id: Int,                // steady
val planName: String,       // steady
val renewalOn: LocalDate    // unstable
)
On this instance, when the renewSubscription known as, the SubscriptionComposable will likely be skipped once more, as a result of now it makes use of the equals() perform as a substitute of ===, which is able to return true in comparison with the earlier state.
When can this happen?
A sensible instance of while you’ll nonetheless must annotate your courses as @Immutable is while you use entities coming from the peripherals of your system, similar to database entities, API entities, Firestore modifications, or others.
As a result of these entities are parsed each time from the underlying knowledge, they create new cases each time. Subsequently, with out the annotation, they’d recompose.
Notice: Recomposing might be sooner than calling
equals()on each parameter. It’s best to all the time measure the impact of your modifications when optimizing stability.
For courses that aren’t a part of your codebase, our steerage was that the one solution to stabilize them is wrapping the category with a category that’s a part of your codebase and annotate that class as @Immutable as a substitute.
Contemplate an instance, the place you’d have a composable that immediately accepts the java.time.LocalDate parameter:
@Composable
enjoyable LatestChangeOn(up to date: LocalDate) {
// current the day parameter on display
}
When you name the renewSubscription perform to replace the newest change, you’ll find yourself in the same scenario as earlier than — the LatestChangeOn composable retains recomposing, regardless if it’s the identical day or not. Nevertheless, there’s no chance of annotating that class on this scenario, as a result of it’s a part of the usual library.
To repair this, you’ll be able to allow a stability configuration file, which might comprise courses or patterns of courses that will likely be thought of steady by the Compose compiler.
To allow it, add stabilityConfigurationFile to the composeCompiler configuration:
composeCompiler {
...// Set path of the config file
stabilityConfigurationFile = rootProject.file("stability_config.conf")
}
And create the stability_config.conf file within the root folder of your venture, wherein you add the LocalDate class:
// add the immutable courses outdoors of your codebase
java.time.LocalDate// alternatively you'll be able to stabilize all java.time courses with *
java.time.*
Stabilize your area mannequin courses
Along with courses that aren’t a part of your codebase, the soundness configuration file might be useful for stabilizing all of your knowledge or area mannequin courses (assuming they’re immutable). This manner, the area module generally is a Java Gradle module and doesn’t want dependency on the Compose compiler.
// stabilize all courses in mannequin package deal
com.instance.app.area.mannequin.*
Remember that annotating a mutable class with the @Immutable annotation, or including the category to the soundness configuration file, generally is a supply of bugs in your codebase, as a result of the Compose compiler is not able to verifying the contract and it would present up as one thing is not recomposing while you suppose it ought to.
One different good thing about sturdy skipping is that it “remembers” all lambdas utilized in composition, even those with unstable captures. Beforehand, lambdas that have been utilizing an unstable class, for instance a ViewModel, may’ve been the reason for recomposition. One of many frequent workarounds was remembering the lambda features.
So, in case you have lambdas wrapped with keep in mind in your codebase, you’ll be able to safely take away the keep in mind name, as a result of it’s achieved mechanically by the Compose compiler:
Display(
-removeItem = keep in mind(viewModel){ { id -> viewModel.removeItem(id) } }
+removeItem = { id -> viewModel.removeItem(id) }
)
The kotlinx.collections.immutable collections like ImmutableList might’ve been used prior to now to make a Listing of things steady and thus stopping a composable from recomposing. In case you have them in your codebase purely for the aim of stopping recompositions of composables with Listing parameters, you may contemplate refactoring them to a daily Listing and add java.util.Listing into the soundness configuration file.
However!
When you try this, your composable may be slower than if the Listing parameter was unstable!
Including Listing to the soundness configuration file means the Listing parameter is in contrast with the equals name, which ultimately results in calling equals on each single merchandise of that checklist. Within the context of a lazy checklist, the identical equals test is then known as once more from the attitude of the merchandise composable, which leads to calculating the equals() name twice for lots of the seen objects, and presumably needlessly for all of the objects that aren’t seen!
If the composable containing the Listing parameter doesn’t have many different UI parts, recomposing it may be sooner than calculating the equals() test.
Nevertheless, there’s nobody dimension suits all strategy right here, so it is best to confirm your selection with benchmarks!
By enabling sturdy skipping mode in your code base, you’ll be able to scale back the necessity to manually craft courses to be steady. Remember that in some circumstances, they nonetheless want handbook crafting, however this may now be simplified with the soundness configuration file!
We hope all of those modifications will simplify the psychological load of fascinated by stability in Compose.
Need extra? See our codelab on sensible efficiency downside fixing in Compose.
The code snippets on this weblog have the next license:
// Copyright 2024 Google LLC. SPDX-License-Identifier: Apache-2.0


