Reputation: 2690
In Jetpack compose, we have the option of using rememberCoroutineScope()
as well as using the LaunchedEffect
composable in order to use coroutines / run suspend functions (show snackbars etc).
The convention I've adopted so far is to remember a single coroutine scope at the top of my compose tree, and pass it down via function arguments to places where it is needed. This vaguely seems like a good practice, but on the other hand it's adding extra noise to my function signatures.
LaunchedEffect
over rememberCoroutineScope()
inside composable functions?rememberCoroutineScope()
in each function where a coroutine is actually launched?Upvotes: 81
Views: 40898
Reputation: 2651
LaunchedEffect
is launched once when the initial composition takes place and It's protected to be launched over and over again against number of potential recompositions (unless it's parameterised and one of the parameters happened to change). Therefore it's good for running animations, snackbar, etc.
LaunchedEffect
is also Composable itself and launches a coroutine behind the scenes.
rememberCoroutineScope
also gives you a coroutine too but it's not a Composable. And that's for a good reason. Otherwise, it would run on every recomposition regardless. Therefore it's usable, for example, within event callbacks (i.e.: onClick) to run suspend
functions.
Upvotes: 0
Reputation: 7603
LaunchedEffect: run suspend functions in the scope of a composable
To call suspend functions safely from inside a composable, use the LaunchedEffect composable. When LaunchedEffect enters the Composition, it launches a coroutine with the block of code passed as a parameter. The coroutine will be cancelled if LaunchedEffect leaves the composition.
rememberCoroutineScope: obtain a composition-aware scope to launch a coroutine outside a composable
As LaunchedEffect is a composable function, it can only be used inside other composable functions. In order to launch a coroutine outside of a composable, but scoped so that it will be automatically canceled once it leaves the composition, use rememberCoroutineScope
More from here.
Upvotes: 8
Reputation: 34
LaunchedEffect as a composable function can only be used inside of another composable function; on the other hand, rememberCoroutineScope allows you to launch a coroutine outside of a composable function
Check this link for further details: https://developer.android.com/jetpack/compose/side-effects
Upvotes: -1
Reputation: 1398
LaunchedEffect launches a coroutine every time it faces new keys. It's a composable so it's bound to the tree and will be destroyed and cancelled when removed from tree.
rememberCoroutineScope creates a coroutine scope bound to the composable. Then you have a Kotlin coroutine scope that you can use anywhere to launch coroutines. The scope will be destroyed when the composable that called it is removed from tree.
Where you call rememberCoroutineScope and where you use the created scope, depends on your needs and the required lifecycle for the coroutines you launch.
Upvotes: 0
Reputation: 1827
Use rememberCoroutineScope() when you are using coroutines and need to cancel and relaunch the coroutine after an event
Use LaunchedEffect() when you are using coroutines and need to cancel and relaunch the coroutine every time your parameter changes and it isn’t stored in a mutable state.
Added code here.
val scope = rememberCoroutineScope()
Row(modifier = Modifier
.clickable { scope.launch { /* call suspend */ } } {
LaunchedEffect(key1 = key){
/* call suspend when entering or key changed */
}
}
An interesting thing is I saw many lines of code as the below.
val scope = rememberCoroutineScope()
Row(modifier = Modifier){
LaunchedEffect(key1 = key){
scope.launch{
/* call suspend when entering or key changed */
}
}
}
However, I don't think we need an extra scope in this case.
Row(modifier = Modifier){
LaunchedEffect(key1 = key){
/* call suspend when entering or key changed */
}
}
Upvotes: 14
Reputation: 1541
Question 1:
LaunchEffect https://developer.android.com/jetpack/compose/side-effects#launchedeffect
rememberCoroutineScope https://developer.android.com/jetpack/compose/side-effects#remembercoroutinescope
Upvotes: -3
Reputation: 66526
rememberCoroutineScope
is a composable function that returns a CoroutineScope bound to the point of the Composition where it's called. The scope will be cancelled when the call leaves the Composition. If you create your own coroutineScope instead of remember you might get MonotonicFrameClock is not available in this CoroutineContext
error as in question here.
LaunchedEffect
is a remember under the hood with coroutineScope which runs when it enters composition and when any of its keys change.
@Composable
@NonRestartableComposable
@OptIn(InternalComposeApi::class)
fun LaunchedEffect(
key1: Any?,
block: suspend CoroutineScope.() -> Unit
) {
val applyContext = currentComposer.applyCoroutineContext
remember(key1) { LaunchedEffectImpl(applyContext, block) }
}
LaunchedEffect is good for launching animations, snackbar, scolls any default suspending function Compose has also another very useful usages is triggering some events without user interaction
for instance executing a callback only when reaching a certain state without user interactions as in this question
LaunchedEffect(statementsByYear != null) {
if (statementsByYear != null) {
onDataAcquired(statementsByYear!!)
}
}
LaunchedEffect gets triggered on composition but as in question since value is null if block condition is not met then when statementsByYear
is set by api statementsByYear != null
changes from true to false and if block condition is met and you run the statement.
Or only once inside a Composable using help of ViewModel saving a flag as in this question.
block of LaunchedEffect(keys) is invoked on composition and when any keys change. If you set keys from your ViewModel this LaunchedEffect will be launched and you can create a conditional block that checks same flag to be true that is contained in ViewModel
LaunchedEffect(mViewModel.isLaunched) {
if(!mViewModel.isLaunched) {
mViewMode.iniBilling(context as Activity)
mViewMode.isLaunched = true
}
}
Also conditional blocks enter composition when conditions are met so by wrapping LaunchedEffect with if block you can define when it will enter and exit composition which means canceling job its coroutineScope might be running as in this answer
if (count > 0 && count <5) {
// `LaunchedEffect` will cancel and re-launch if
// `scaffoldState.snackbarHostState` changes
LaunchedEffect(scaffoldState.snackbarHostState) {
// Show snackbar using a coroutine, when the coroutine is cancelled the
// snackbar will automatically dismiss. This coroutine will cancel whenever
// if statement is false, and only start when statement is true
// (due to the above if-check), or if `scaffoldState.snackbarHostState` changes.
scaffoldState.snackbarHostState.showSnackbar("count $count")
}
}
This block will enter composition when count is bigger than 0 and stay in composition while count is less than 5 but since it's LaunchedEffect it will trigger once but if count reaches 5 faster than Snackbar duration Snackbar gets canceled because block leaves composition.
Upvotes: 4
Reputation: 23844
Leaving my understanding here:
Question 1:
LaunchedEffect
should be used when you want that some action must be taken when your composable is first launched/relaunched (or when the key parameter has changed). For example, when you want to request some data from your ViewModel or run some sort of animation...
rememberCoroutineScope
on the other hand, is specific to store the Coroutine scope allowing the code to launch some suspend
function...
imho, the only relation between them is that you can also use a LaunchedEffect
to launch a coroutine...
Question 2: As you can see in the docs, rememberCoroutineScope
will keep the reference of the coroutine's scope in a specific point of the composition. Therefore, if a given composable is removed from the recomposition, that coroutine will be cancelled automatically. For instance, you have the following composable calls A -> B -> C
. If you remember the coroutine scope in C
and it is removed from the composition, the coroutine is automatically cancelled. But if you remember from A
, pass the scope through B
and C
, use this scope in C
, and then C
is removed, the coroutine will continue running (because it was remembered in A
)...
Upvotes: 86