In an Android application, you typically need to collect the Kotlin data stream from the UI layer to display the data updates on the screen. At the same time, you want to collect these data streams to avoid unnecessary operations and waste of resources (including CPU and memory), as well as to prevent data leakage when the View enters the background.
. This article will take you to learn how to use LifecycleOwner addRepeatingJob, Lifecycle. RepeatOnLifecycle and Flow. The flowWithLifecycle API to avoid the waste of resources; It will also explain why these APIs are appropriate as the default choices for collecting data streams at the UI layer.
Waste of resources
Regardless of the specific implementation of the data Flow producer, it is recommended that the Flow
API be exposed from the lower levels of the application. However, you should also ensure that data flow collection operations are secure.
Using some existing apis (such as CoroutineScope launch, Flow < T > launchIn or LifecycleCoroutineScope launchWhenX) collection based on the channel or use with buffer operators (such as Data from the cold stream of Buffer, Conflate, Flowon, or Sharein) is unsafe unless you manually canceled the Job that started the coroutine when the Activity entered the background. These APIs keep them active while internal producers send items to buffers in the background, which wastes resources.
Note:Cold flowIs a type of data flow that executes a producer’s block of code on demand as new subscribers collect data.
For example, in the following example, we use CallbackFlow to send a data stream of location updates:
// Channel-based implementation of cold flow, Can send the Location update fun FusedLocationProviderClient. LocationFlow () = callbackFlow < Location > {val callback = object: LocationCallback() { override fun onLocationResult(result: LocationResult?) { result ? : return try { offer(result.lastLocation) } catch(e: Exception) {} } } requestLocationUpdates(createLocationRequest(), callback, Looper. GetMainLooper ()). AddOnFailureListener {e-> Close (E) // Close the Flow in case of an exception} // Clean up at the end of the Flow collection awaitClose { removeLocationUpdates(callback) } }
Note: CallbackFlow is used internallychannelImplement, its concept versus blockingThe queue) is very similar and defaults to 64.
Collecting this data stream from the UI layer using any of the aforementioned APIs will cause it to continue sending location information, even if the view no longer presents the data! Here is an example:
class LocationActivity : AppCompatActivity() { override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState) // First collects data from the data stream when the View is STARTED, and SUSPENDS the collection operation when the life cycle is STOPPED. // Cancel the collection of the data stream while the View is in the Destroyed state. LifecycleScope. LaunchWhenStarted {locationProvider. LocationFlow () collect {/ / new location! Update map}} // The same problem exists with: / / - lifecycleScope. Launch {/ * from locationFlow here () collect data * /} / / - locationProvider locationFlow () onEach {/ *... */ }.launchIn(lifecycleScope) } }
LifecycleScope. LaunchWhenStarted suspends the execution of coroutines. Although the new location information is not processed, the CallbackFlow producer continues to send location information. Using the Lifecyclescope. Launch or LaunchIn API is even more dangerous because the view will continue to consume location information, even when it’s in the background! This situation can cause your application to crash.
To address the problems posed by these APIs, you need to manually cancel the collection when the view goes into the background to cancel CallbackFlow and avoid the location provider from constantly sending items and wasting resources. For example, you could do something like this:
Class LocationActivity: AppCompatActivity() {private Var LocationUpdatesJob: Job? = null override fun onStart() { super.onStart() locationUpdatesJob = lifecycleScope.launch { LocationProvider. LocationFlow (). Collect {/ / new location! Update the map. }}} Override fun onStop() {// Stop collecting data when the view enters the background LocationUpdatesJob? .cancel() super.onStop() } }
It’s a good solution, but it’s a bit long. If there’s one universal truth about Android developers in the world, it’s that we all hate to write template code. One of the biggest benefits of not having to write template code is that the less code you write, the less likely it is to go wrong!
LifecycleOwner.addRepeatingJob
Now that we’re in the same boat and we know what the problem is, it’s time to find a solution. Our solution requires: 1. Simplicity; 2. 2. Friendly or easy to remember and understand; What’s more, 3. Safety! Regardless of the implementation details of the data flow, it should be able to handle all use cases.
Before, you should use the API is the lifecycle – runtime – LifecycleOwner provided KTX repository. AddRepeatingJob. Please refer to the following code:
class LocationActivity : AppCompatActivity() { override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState) // The collection of data from the data stream is STARTED when the View is in the STARTED state and STOPPED when the // life cycle is in the STOPPED state. // It will automatically start data collection when the life cycle starts again in the Started state. LifecycleOwner. AddRepeatingJob (Lifecycle. State. STARTED) {locationProvider. LocationFlow () collect {/ / new location! Update the map}}}}
AddrePeatingJob takes Lifecycle.State as an argument and uses it, along with the passed code block, to automatically create and start a new coroutine when the Lifecycle reaches that State; It also cancels running coroutines if their lifetime falls below that state.
Because AddrePeatingJob automatically cancels the coroutine when it is no longer needed, you can avoid generating template code associated with the cancellation operation. As you might have guessed, in order to avoid unexpected behavior, this API needs to be called in the onCreate method of the Activity or the onViewCreated method of the Fragment. Here’s an example with a Fragment:
class LocationFragment: Fragment() { override fun onViewCreated(view: View, savedInstanceState: Bundle?) {/ /... ViewLifecycleOwner. AddRepeatingJob (Lifecycle. State. STARTED) {locationProvider. LocationFlow () collect {/ / new location! Update the map}}}}
Note: These APIs are in
Lifecycle: lifecycle - runtime - KTX: 2.4.0 - alpha01
Library or a newer version of it.
Using repeatOnLifecycle
And save for a more flexible API calls the CoroutineContext purpose, we also provide Lifecycle. The suspended function repeatOnLifecycle for your use. RepeatOnLifecycle suspends the coroutine that called it and reexecutes the block of code as it goes in and out of the target state, and finally resumes the coroutine that called it when Lifecycle goes into the destruction state.
If you need to perform a configuration task before a rework and you want the task to remain suspended until the rework begins, this API can help you do so. Here is an example:
class LocationActivity : AppCompatActivity() { override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(SavedInstanceState) Lifecyclescope. Launch {val ExpensiveObject = createExpensiveObject() Lifecycle. RepeatOnLifecycle lifecycle. State. (STARTED) {/ / into a State in the lifecycle of STARTED to repetitive tasks, // Operate on ExpensiveObject on STOPED} // When the coroutine resumes, 'lifecycle' is in the DESTROY state. RepeatOnLifecycle will suspend the execution of the coroutine before it enters the Destroyed state.
Flow.flowWithLifecycle
The flow.flowWithLifecycle operator can also be used when you only need to collect one data stream. Also within this API using suspend Lifecycle. RepeatOnLifecycle function realization, and will enter and leave the life cycle of the target state sent items and to eliminate internal producer.
class LocationActivity : AppCompatActivity() { override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) locationProvider.locationFlow() .flowWithLifecycle(this, Lifecycle.State. Started).oneach {// New location! Update map}.launchin (Lifecyclescope)}}
Note:
Flow.flowWithLifecycle
The API is named afterFlow.flowOn(CoroutineContext)
Is a precedent, because it will modify the data stream that collects the upstream data stream without affecting the downstream data streamCoroutineContext
. withflowOn
Another similarity is that,Flow.flowWithLifecycle
Buffers have also been added to prevent consumers from failing to keep up with producers. This feature stems from the use of thecallbackFlow
.
Configuring internal producers
Even if you use these APIs, be careful of heat flows that may waste resources, even if they are not collected! While there are some appropriate use cases for these heat flows, it is important to take care of them and document them when necessary. On the other hand, there are situations where keeping the internal data flow producers active in the background can benefit some use cases, even though it may be wasteful of resources, such as when you need to refresh the available data on the spot, rather than retrieve and temporarily display stale data. Depending on the use case, you can decide whether the producer needs to be always active.
You can control them by using the exposed subscriptionCount fields in both the MutableStateflow and MutableSharedFlow APIs, where the internal producer stops when the value of this field is 0. By default, objects holding data flow instances remain producer active as long as they are in memory. There are also some appropriate use cases for these APIs, such as using Stateflow to expose the UiState from the ViewModel to the UI. This is appropriate because it means that the ViewModel always needs to provide the latest UI state to the View.
Similarly, you can configure the Flow.stateIn and Flow.shareIn operators for such operations using a shared start policy. Whilesubscribe () stops the internal producers if there are no active subscribers! Correspondently, producers within the data streams will remain active as long as the CoroutineScope used by them is active, regardless of whether the data streams are active or Lazily.
Note: The APIs described in this article are good as the default way to collect data flows from the UI, and should be used regardless of how the data flows are implemented. These APIs do what they’re meant to do: stop collecting data streams from the UI when it’s not visible on the screen. Whether or not the data flow should always be active depends on its implementation.
Securely collect data streams in Jetpack Compose
The flow. collectAsState function collects data streams from the composable in the Compose and can represent the value as State
to update the Compose UI. Even if the Compose is not reconstituted while the host Activity or Fragment is in the background, the data stream producer remains active and wastes resources. Compose may experience the same problems as the View system.
When collecting a data stream in the Compose, you can use the flow.flowwithlifecycle operator, as shown in the following example:
@Composable fun LocationScreen(locationFlow: Flow<Flow>) { val lifecycleOwner = LocalLifecycleOwner.current val locationFlowLifecycleAware = remember(locationFlow, lifecycleOwner) { locationFlow.flowWithLifecycle(lifecycleOwner.lifecycle, Lifecycle. State. STARTED)} val location by locationFlowLifecycleAware. CollectAsState () / / the current location, you can take it to do some operation}
Note that you need to remember that the lifecycle-aware data flow uses LocationFlow and LifecycleOwner as keys so that the same data flow is always used unless one of the keys changes.
Compose (Side – effect) is a must in the Side effects of a controlled environment, therefore, use LifecycleOwner. AddRepeatingJob unsafe. Instead, you can use LauncheDefect to create coroutines that follow the Composable life cycle. In its code block, if you need to host life cycle in a State to perform a code block, can hang function called Lifecycle. RepeatOnLifecycle.
Contrast LiveData
You might think that these APIs behave a lot like Livedata — and they do! LiveData is aware of Lifecycle, and its restart behavior makes it ideal for observing data streams from the UI. Similarly LifecycleOwner addRepeatingJob, suspend Lifecycle. RepeatOnLifecycle and Flow. The flowWithLifecycle etc API as well.
In a pure Kotlin application, using these APIs is a very natural alternative to Livedata for collecting data streams. If you use these APIs to collect data flows, there is no additional benefit to switching to Livedata (as opposed to using coroutines and flows). And since Flow can collect data from any Dispatcher, it can also pass through itThe operatorGet more functionality, so Flow is more flexible. Livedata has a relatively limited number of available operators, and it always looks at data from the UI thread.
Data binding support for Stateflow
On the other hand, one of the reasons you might want to use Livedata is that it is supported by data binding. But Stateflow is the same! For more information on data binding support for Stateflow, see the official documentation.
In Android development, Please use the LifecycleOwner addRepeatingJob, suspend Lifecycle. RepeatOnLifecycle or Flow. FlowWithLifecycle safely collected data stream from the UI layer.