Ciclo de vida de la actividad

Cuando un usuario navega por tu app, sale de ella y vuelve a entrar, las instancias de Activity de tu app pasan por diferentes estados de su ciclo de vida. La clase Activity proporciona una serie de devoluciones de llamada que permiten a la actividad saber cuándo cambia un estado o que el sistema está creando, deteniendo o reanudando una actividad, o destruyendo el proceso en el que se encuentra la actividad.

Dentro de los métodos de devolución de llamada de ciclo de vida, puedes declarar el comportamiento que tendrá tu actividad cuando el usuario la abandone y la reanude. Por ejemplo, si creas un reproductor de video en streaming, puedes pausar el video y cancelar la conexión de red cuando el usuario cambia a otra app. Cuando el usuario vuelve, puedes volver a establecer la conexión con la red y permitir que el usuario reanude el video desde el mismo punto.

Cada devolución de llamada te permite realizar un trabajo específico que es apropiado para un cambio de estado determinado. Hacer el trabajo preciso en el momento adecuado y administrar las transiciones correctamente hace que tu app sea más sólida y eficiente. Por ejemplo, una buena implementación de las devoluciones de llamada del ciclo de vida puede ayudar a tu app a evitar lo siguiente:

  • No falle si el usuario recibe una llamada telefónica o cambia a otra app mientras usa la tuya.
  • No consuma recursos valiosos del sistema cuando el usuario no la use de forma activa.
  • No pierda el progreso del usuario si este abandona tu app y regresa a ella posteriormente.
  • No falle ni pierda el progreso del usuario cuando se gire la pantalla entre la orientación horizontal y la vertical.

En este documento, se explica en detalle el ciclo de vida de las actividades. Al principio del documento, se describe el paradigma del ciclo de vida. A continuación, se explica cada una de las devoluciones de llamada: qué sucede internamente mientras se ejecutan y qué se debe implementar durante ellas.

Luego, se presenta de forma breve la relación entre el estado de la actividad y la vulnerabilidad de un proceso que el sistema está por finalizar. Por último, se abordan varios temas relacionados con las transiciones entre los estados de una actividad.

Para obtener información sobre cómo controlar los ciclos de vida, incluida orientación sobre prácticas recomendadas, consulta Ciclo de vida en Jetpack Compose y Cómo guardar estados de IU. Para aprender a diseñar una app sólida y de calidad utilizando actividades junto con componentes de arquitectura, consulta la Guía de arquitectura de apps.

Conceptos de los ciclos de vida de las actividades

Para navegar por las transiciones entre las etapas del ciclo de vida de la actividad, la clase Activity proporciona un conjunto básico de seis devoluciones de llamada: onCreate, onStart, onResume, onPause, onStop y onDestroy. El sistema invoca cada una de estas devoluciones de llamada cuando la actividad entra en un nuevo estado.

En la figura 1, se muestra una representación visual de este paradigma.

Figura 1: Ilustración simplificada del ciclo de vida de la actividad.

Cuando el usuario comienza a abandonar la actividad, el sistema llama a métodos para desmantelarla. En algunos casos, la actividad solo se desmantela parcialmente y aún reside en la memoria, como cuando el usuario cambia a otra app. En estos casos, la actividad aún puede volver al primer plano.

Si el usuario regresa a la actividad, se reanuda desde donde la dejó. Con algunas excepciones, se restringen las apps para que no inicien actividades cuando se ejecutan en segundo plano.

La probabilidad de que el sistema finalice un proceso determinado, junto con las actividades que contiene, depende del estado de la actividad en ese momento. Para obtener más información sobre la relación entre el estado y la vulnerabilidad a la expulsión, consulta la sección sobre el estado de la actividad y la expulsión de memoria.

Según la complejidad de tu actividad, es probable que no necesites implementar todos los métodos del ciclo de vida. Sin embargo, es importante que comprendas cada uno de ellos y que implementes aquellos que garanticen que tu app se comporte como esperan los usuarios.

Compose y el ciclo de vida

En Compose, evita colocar la lógica empresarial o la configuración manual del observador directamente dentro de las devoluciones de llamada de la actividad, como onStart o onResume. En su lugar, usa efectos optimizados para Lifecycle y observadores con estado que se alineen automáticamente con la presencia de la IU en la pantalla.

  • Recopilación optimizada para ciclos de vida: Usa collectAsStateWithLifecycle para consumir flujos de tu ViewModel. Esta API comienza a recopilar datos automáticamente cuando la IU ingresa al estado Started y se detiene cuando pasa a segundo plano, lo que evita el consumo innecesario de recursos. Después de recopilar el flujo como estado, puedes usar LifecycleEffects para ejecutar código cuando se produce un evento de ciclo de vida.
  • Flujo de lógica: Cuando se usan estas APIs, la IU reacciona al estado del ciclo de vida de forma natural a través del árbol de composición, lo que garantiza que la lógica empresarial solo se ejecute cuando el usuario interactúa de forma activa con el componente.

Para obtener más información sobre Compose y el ciclo de vida, consulta Ciclo de vida en Jetpack Compose.

Devoluciones de llamada del ciclo de vida

En esta sección, se brinda información conceptual y de implementación sobre los métodos de devolución de llamada utilizados durante el ciclo de vida de la actividad.

Algunas acciones pertenecen a los métodos del ciclo de vida de la actividad. Sin embargo, coloca el código que implementa las acciones de un componente dependiente en el componente, en lugar de en el método del ciclo de vida de la actividad. Para ello, debes hacer que el componente dependiente priorice el ciclo de vida. Para obtener información sobre cómo hacer que los componentes de tus dependencias prioricen el ciclo de vida, consulta Ciclo de vida en Jetpack Compose.

onCreate

Debes implementar esta devolución de llamada, que se activa cuando el sistema crea la actividad por primera vez. Cuando se crea la actividad, esta entra en el estado Created. En el método onCreate, ejecuta la lógica de arranque básica de la aplicación que ocurre solo una vez en toda la vida de la actividad.

Por ejemplo, tu implementación de onCreate podría vincular datos a listas, asociar la actividad con un ViewModel y crear instancias de algunas variables de alcance de clase. Este método recibe el parámetro savedInstanceState, que es un objeto Bundle que contiene el estado ya guardado de la actividad. Si la actividad nunca existió, el valor del objeto Bundle es nulo.

Si tienes un componente que prioriza el ciclo de vida y que está conectado al ciclo de vida de tu actividad, recibirá el evento ON_CREATE. Se llama al método anotado con @OnLifecycleEvent para que tu componente que prioriza el ciclo de vida pueda realizar cualquier código de configuración que necesite para el estado de creación.

En el siguiente ejemplo, se muestra cómo integrar un elemento Text componible en una actividad mínima:

class ExampleActivity : ComponentActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)

        setContent { // In here, we can call composables!
            MaterialTheme {
                Greeting(name = "compose")
            }
        }
    }
}

@Composable
fun Greeting(name: String) {
    Text(text = "Hello $name!")
}

Tu actividad no permanece en el estado Created. Después de que se termina de ejecutar el método onCreate, la actividad entra en el estado Started, y el sistema llama rápidamente a los métodos onStart y onResume.

onStart

Cuando la actividad entra en el estado Started, el sistema invoca onStart. Esta llamada hace que el usuario pueda ver la actividad mientras la app se prepara para que esta entre en primer plano y se convierta en interactiva. Por ejemplo, este método es donde se inicializa el código que mantiene la IU.

Cuando la actividad pasa al estado Started, cualquier componente que priorice el ciclo de vida vinculado al de la actividad recibe el evento ON_START.

El método onStart se completa rápidamente y, al igual que con el estado Created, la actividad no permanece en el estado Started. Una vez finalizada esta devolución de llamada, la actividad entra en el estado Resumed y el sistema invoca el método onResume.

onResume

Cuando la actividad entra en el estado Resumed, pasa al primer plano y el sistema invoca la devolución de llamada onResume. Este es el estado en el que la app interactúa con el usuario. La app permanece en este estado hasta que ocurre algún evento que la quita de foco, como que el dispositivo reciba una llamada telefónica, que el usuario navegue a otra actividad o que se apague la pantalla del dispositivo.

Cuando la actividad pasa al estado Resumed, cualquier componente que priorice el ciclo de vida vinculado al de la actividad recibe el evento ON_RESUME. Aquí es donde los componentes del ciclo de vida pueden habilitar cualquier funcionalidad que necesite ejecutarse mientras el componente esté visible y en primer plano, como, por ejemplo, iniciar una vista previa de la cámara.

Cuando se produce un evento de interrupción, la actividad entra en el estado Paused y el sistema invoca la devolución de llamada onPause.

Si la actividad regresa al estado Resumed desde Paused, el sistema volverá a llamar al método onResume. Por este motivo, implementa onResume para inicializar los componentes que liberes durante onPause y para realizar otras inicializaciones que deban ejecutarse cada vez que la actividad entre en el estado Resumed.

A continuación, se incluye un ejemplo de un componente que prioriza el ciclo de vida que accede a la cámara cuando el componente recibe el evento ON_RESUME:

class CameraComponent : LifecycleObserver {
    ...
    @OnLifecycleEvent(Lifecycle.Event.ON_RESUME)
    fun initializeCamera() {
        if (camera == null) {
            getCamera()
        }
    }
    ...
}

El código anterior inicializa la cámara una vez que LifecycleObserver recibe el evento ON_RESUME. Sin embargo, en el modo multiventana, tu actividad puede ser totalmente visible incluso cuando se encuentra en el estado Paused. Por ejemplo, si la app está en el modo multiventana y el usuario presiona la ventana que no contiene tu actividad, esta se moverá al estado Paused.

Si deseas que la cámara esté activa solo cuando se reanude la app (visible y activa en primer plano), inicializa la cámara después del evento ON_RESUME que se mostró antes. Si deseas mantener la cámara activa mientras la actividad está en el estado Paused, pero visible, como en el modo multiventana, inicializa la cámara después del evento ON_START.

Sin embargo, tener la cámara activa mientras la actividad se encuentra en el estado Paused puede impedir que otra app que se encuentre en el estado Resumed pueda acceder a la cámara en el modo multiventana. En ocasiones, es necesario mantener la cámara activa mientras la actividad está en el estado Paused, pero esto podría degradar la experiencia general del usuario.

Por este motivo, piensa con cuidado en qué parte del ciclo de vida es más adecuado tomar el control de los recursos compartidos del sistema en el contexto del modo multiventana. Para obtener más información sobre la compatibilidad con el modo multiventana, consulta Compatibilidad con el modo multiventana.

Independientemente del evento en el que decidas realizar una operación de inicialización, asegúrate de utilizar el evento de ciclo de vida correspondiente para liberar el recurso. Si inicializas algún evento después de ON_START, libéralo o finalízalo después del evento ON_STOP. Si inicializas un evento después del evento ON_RESUME, libéralo después del evento ON_PAUSE.

El fragmento de código anterior coloca el código de inicialización de la cámara en un componente que prioriza el ciclo de vida. En su lugar, puedes poner este código directamente en las devoluciones de llamada del ciclo de vida de la actividad, como onStart y onStop, pero no lo recomendamos. Agregar esta lógica a un componente independiente que prioriza el ciclo de vida te permite reutilizar el componente en varias actividades sin tener que duplicar el código. Para obtener información sobre cómo crear un componente que priorice el ciclo de vida, consulta Ciclo de vida en Jetpack Compose.

onPause

El sistema llama a este método como la primera indicación de que el usuario está abandonando tu actividad, aunque no siempre significa que se está destruyendo la actividad. Indica que la actividad ya no está en primer plano, pero sigue siendo visible si el usuario está en el modo multiventana. Hay varios motivos por los que una actividad podría entrar en este estado:

  • Un evento que interrumpe la ejecución de la app, como se describe en la sección sobre la devolución de llamada onResume, pausa la actividad actual. Este es el caso más común.
  • En el modo multiventana, solo una app tiene el foco en cualquier momento, y el sistema pausa todas las demás.
  • La apertura de una nueva actividad semitransparente, como un diálogo, pausa la actividad que cubre. Mientras la actividad siga siendo parcialmente visible, pero no esté en primer plano, se mantendrá pausada.

Cuando una actividad pasa al estado Paused, cualquier componente que priorice el ciclo de vida vinculado al ciclo de vida de la actividad recibe el evento ON_PAUSE. Aquí es donde los componentes del ciclo de vida pueden detener cualquier funcionalidad que no necesite ejecutarse mientras el componente no esté en primer plano, como detener una vista previa de la cámara.

Utiliza el método onPause para pausar o ajustar las operaciones que no pueden continuar, o que podrían continuar con moderación, mientras Activity se encuentra en estado Paused y que esperas reanudar en breve.

También puedes usar el método onPause para liberar recursos del sistema, controladores de sensores (como el GPS) o cualquier otro recurso que afecte la duración de la batería mientras tu actividad esté en pausa y el usuario no los necesite.

Sin embargo, como se mencionó en la sección sobre onResume, una actividad con el estado Paused puede ser completamente visible si la app está en el modo multiventana. Considera usar onStop en lugar de onPause para liberar o ajustar por completo los recursos y operaciones relacionados con la IU a fin de admitir mejor el modo multiventana.

En el siguiente ejemplo de un LifecycleObserver que reacciona ante el evento ON_PAUSE, que es la contrapartida del ejemplo del evento ON_RESUME anterior, se libera la cámara que se inicializa después de recibir el evento ON_RESUME:

class CameraComponent : LifecycleObserver {
    ...
    @OnLifecycleEvent(Lifecycle.Event.ON_PAUSE)
    fun releaseCamera() {
        camera?.release()
        camera = null
    }
    ...
}

En este ejemplo, se coloca el código de liberación de la cámara después de que el LifecycleObserver recibe el evento ON_PAUSE.

La ejecución de onPause es muy breve y no necesariamente ofrece tiempo suficiente para realizar operaciones de guardado. Por este motivo, no uses onPause para guardar datos de la aplicación o del usuario, realizar llamadas de red o ejecutar transacciones de base de datos. Es posible que este trabajo no se complete antes de que se complete el método.

En su lugar, realiza operaciones de finalización de cargas pesadas durante onStop. Para obtener más información sobre las operaciones adecuadas que se pueden realizar durante onStop, consulta la siguiente sección. Para obtener más información sobre cómo guardar datos, consulta la sección sobre cómo guardar y restablecer el estado.

La finalización del método onPause no significa que la actividad abandone el estado Paused. Más bien, la actividad permanecerá en ese estado hasta que se reanude o se vuelva completamente invisible para el usuario. Si se reanuda la actividad, el sistema volverá a invocar la devolución de llamada onResume.

Si la actividad regresa del estado Paused al estado Resumed, el sistema mantendrá la instancia Activity residente en la memoria y la volverá a llamar cuando invoque onResume. En esta situación, no es necesario que reinicialices los componentes que se crearon durante los métodos de devolución de llamada que llevan al estado Resumed. Si la actividad se vuelve completamente invisible, el sistema llamará a onStop.

onStop

Cuando el usuario ya no puede ver tu actividad, esta entra en el estado Stopped y el sistema invoca la devolución de llamada onStop. Esto puede ocurrir cuando una actividad recién lanzada cubre toda la pantalla. El sistema también llama a onStop cuando la actividad termina de ejecutarse y está a punto de finalizar.

Cuando la actividad pasa al estado Stopped, cualquier componente que priorice el ciclo de vida vinculado al de la actividad recibe el evento ON_STOP. Aquí es donde los componentes del ciclo de vida pueden detener cualquier funcionalidad que no necesite ejecutarse mientras el componente no sea visible en la pantalla.

En el método onStop, libera o ajusta los recursos que no son necesarios mientras la app no sea visible para el usuario. Por ejemplo, tu app podría pausar animaciones o cambiar de actualizaciones de ubicación detalladas a más generales. Usar onStop en lugar de onPause significa que el trabajo relacionado con la IU continúa, incluso cuando el usuario está viendo tu actividad en el modo multiventana.

Además, usa onStop para realizar operaciones de finalización con un uso relativamente intensivo de la CPU. Por ejemplo, si no encuentras un momento más oportuno para guardar información en una base de datos, puedes hacerlo en onStop. Por ejemplo, a continuación, se muestra una implementación de onStop que guarda los contenidos del borrador de una nota en el almacenamiento persistente:

override fun onStop() {
    super.onStop()

    // Delegate the save operation to the ViewModel, which handles the
    // background thread operations (e.g., using Kotlin Coroutines and Room).
    noteViewModel.saveDraft()
}

Cuando tu actividad entra en el estado Stopped, se mantiene el objeto Activity en la memoria. Mantiene toda la información de estado y de miembros, pero no está vinculado al administrador de ventanas. Cuando se reanuda la actividad, esta recuerda la información.

Desde el estado Stopped, la actividad regresa a interactuar con el usuario o se termina de ejecutar y desaparece. Si la actividad regresa, el sistema invoca a onRestart. Si se terminó de ejecutar Activity, el sistema llamará a onDestroy.

onDestroy

Se llama a onDestroy antes de que se destruya la actividad. El sistema invoca esta devolución de llamada por uno de los siguientes motivos:

  1. La actividad está terminando, debido a que el usuario la descarta por completo o a que se llama a finish en la actividad.
  2. El sistema está finalizando temporalmente la actividad debido a un cambio de configuración, como la rotación del dispositivo o el ingreso al modo multiventana.

Cuando la actividad pasa al estado Destroyed, cualquier componente que priorice el ciclo de vida vinculado al de la actividad recibe el evento ON_DESTROY. Aquí es donde los componentes del ciclo de vida pueden liberar espacio que necesiten antes de que se destruya el objeto Activity.

En lugar de poner lógica en tu Activity para determinar por qué se destruye, usa un objeto ViewModel para contener los datos de vista relevantes para tu Activity. Si se vuelve a crear el Activity debido a un cambio de configuración, el ViewModel no tiene que hacer nada, ya que se conserva y se entrega a la siguiente instancia de Activity.

Si no se vuelve a crear el objeto Activity, se llama al método onCleared del objeto ViewModel, en el que puede limpiar los datos que necesite antes de que se destruya. Puedes diferenciar estos dos casos con el método isFinishing.

Si la actividad está terminando, onDestroy es la devolución de llamada del ciclo de vida final que recibe la actividad. Si se llama a onDestroy como resultado de un cambio de configuración, el sistema crea inmediatamente una nueva instancia de actividad y luego llama a onCreate en esa nueva instancia en la nueva configuración.

La devolución de llamada onDestroy libera todos los recursos que no liberaron las devoluciones de llamada anteriores, como onStop.

Estado de actividad y expulsión de memoria

El sistema finaliza los procesos cuando necesita liberar RAM. La probabilidad de que el sistema finalice un proceso determinado dependerá del estado del proceso en ese momento. El estado del proceso, a su vez, depende del estado de la actividad que se ejecuta en el proceso. La tabla 1 muestra las correlaciones entre el estado del proceso, el estado de la actividad y la probabilidad de que el sistema finalice el proceso. Esta tabla solo se aplica si un proceso no ejecuta otros tipos de componentes de la aplicación.

Probabilidad de que finalice

Estado del proceso

Estado final de la actividad

Más baja

Primer plano (en foco o por estar en él)

Reanudado

Baja

Visible (sin enfoque)

Iniciada/Pausada

Alto

Segundo plano (invisible)

Detenida

Más alta

Vacío

Finalizado

Tabla 1. Relación entre el ciclo de vida del proceso y el estado de la actividad.

El sistema nunca finaliza una actividad de forma directa para liberar memoria. En su lugar, finaliza el proceso en el que se ejecuta la actividad para eliminar no solo la actividad, sino también todo lo que se ejecuta en el proceso. Para aprender a preservar y restablecer el estado de la IU de tu actividad cuando se produce el cierre del proceso iniciado por el sistema, consulta la sección sobre cómo guardar y restablecer el estado.

El usuario también puede finalizar un proceso utilizando el Administrador de aplicaciones, en Configuración, para finalizar la app correspondiente.

Para obtener más información sobre los procesos, consulta Descripción general de procesos y subprocesos.

Cómo guardar y restablecer el estado transitorio de la IU

El usuario espera que se conserve el estado de la IU de una actividad durante un cambio de configuración, como la rotación o el cambio al modo multiventana. Sin embargo, el sistema finaliza la actividad de forma predeterminada cuando se produce un cambio de configuración de este tipo, lo que elimina cualquier estado de la IU almacenado en la instancia de actividad.

Del mismo modo, un usuario espera que el estado de la IU siga siendo el mismo si cambia temporalmente de tu app a una diferente y, luego, regresa a tu app. Sin embargo, el sistema puede finalizar el proceso de tu aplicación mientras el usuario no la esté utilizando y tu actividad esté detenida.

Cuando las restricciones del sistema destruyen la actividad, conserva el estado transitorio de la IU del usuario con una combinación de ViewModel (para la lógica empresarial compleja y el estado de la pantalla), la API de rememberSaveable de Jetpack Compose (para el estado de la IU ligero) o el almacenamiento local. Para obtener más información sobre las expectativas de los usuarios en comparación con el comportamiento del sistema y sobre la mejor manera de preservar los datos complejos del estado de la IU en toda la actividad iniciada por el sistema y el cierre del proceso, consulta Cómo guardar estados de IU.

rememberSaveable sobrevive automáticamente a los cambios de configuración y al cierre del proceso iniciado por el sistema, ya que agrupa el estado de forma interna, lo que proporciona una experiencia fluida sin necesidad de código repetitivo a nivel de la actividad.

Estado de la instancia

Existen algunas situaciones en las que finaliza tu actividad debido al comportamiento normal de la app, por ejemplo, cuando el usuario presiona el botón Atrás o tu actividad indica su propia finalización llamando al método finish.

Cuando finaliza tu actividad porque el usuario presiona Atrás o la actividad se finaliza a sí misma, se pierde para siempre el concepto de esa instancia Activity del sistema y del usuario. En estos casos, las expectativas del usuario coinciden con el comportamiento del sistema y no tienes trabajo adicional que hacer.

Sin embargo, si el sistema finaliza la actividad debido a restricciones (como un cambio de configuración o presión de memoria), entonces, aunque haya desaparecido la instancia real de Activity, el sistema recuerda que existía. Si el usuario intenta volver a la actividad, el sistema crea una nueva instancia de esa actividad utilizando un conjunto de datos guardados que describen el estado de la actividad cuando finalizó.

Los datos guardados que el sistema utiliza para restablecer el estado anterior se denominan estado de instancia. En segundo plano, es una colección de pares clave-valor. De forma predeterminada, el sistema usa el estado de la instancia para guardar información básica sobre el diseño de la IU, como la entrada de texto del usuario o las posiciones de desplazamiento.

Puedes conectarte a este comportamiento del sistema con rememberSaveable. Si se destruye y se vuelve a crear la instancia de tu actividad, se restablecerá automáticamente cualquier estado de la IU encapsulado en rememberSaveable, sin necesidad de que escribas código adicional a nivel de la actividad.

Sin embargo, es probable que tu actividad tenga información de estado más compleja que desees restablecer, como datos del usuario, respuestas de red o variables de miembro que registren el progreso del usuario. El mecanismo de estado de instancia (y, por extensión, rememberSaveable) no es apropiado para preservar más que una cantidad trivial de datos, ya que requiere serialización en el subproceso principal y consume memoria del proceso del sistema.

Para conservar más que una cantidad muy pequeña de datos, adopta un enfoque combinado con el almacenamiento local persistente, la clase ViewModel y el hoisting de estado de Compose, como se describe en Cómo guardar estados de IU.

Cómo guardar un estado de IU simple y ligero usando rememberSaveable

A medida que comienza a detenerse tu actividad, el sistema se prepara para guardar la información del estado en un paquete de estado de instancia. Para conectarte con este comportamiento del sistema, usa rememberSaveable directamente dentro de tus funciones de componibilidad. rememberSaveable guarda y restablece automáticamente el estado transitorio de la IU, como la entrada de texto del usuario o las posiciones de desplazamiento, en la recreación de la actividad.

Para guardar información de estado personalizada y liviana (como el progreso de un usuario en un juego), declara tu estado con rememberSaveable. El framework de Compose controla la serialización en el paquete de estado de la instancia de forma interna:

var userTypedQuery by rememberSaveable(typedQuery, stateSaver = TextFieldValue.Saver) {
    mutableStateOf(
        TextFieldValue(text = typedQuery, selection = TextRange(typedQuery.length))
    )
}

Para guardar datos persistentes, como las preferencias del usuario o información de una base de datos, aprovecha las oportunidades apropiadas cuando tu actividad esté en primer plano. Si no se presenta tal oportunidad, guarda los datos persistentes durante el método onStop.

Cómo restablecer el estado de la IU de la actividad utilizando el estado de la instancia guardada

Cuando se vuelve a crear tu actividad tras haber finalizado, la restauración del estado es automática. Cuando usas rememberSaveable, no necesitas escribir ninguna lógica de restablecimiento explícita, verificar si hay paquetes nulos ni anular las devoluciones de llamada de actividad. El código que inicializa y guarda tu estado también lo restablece sin problemas cuando la actividad vuelve:

var userTypedQuery by rememberSaveable(typedQuery, stateSaver = TextFieldValue.Saver) {
    mutableStateOf(
        TextFieldValue(text = typedQuery, selection = TextRange(typedQuery.length))
    )
}

Actividades y navegación

Es probable que una app realice la transición entre pantallas muchas veces durante su ciclo de vida, por ejemplo, cuando el usuario presiona el botón Atrás del dispositivo o selecciona un nuevo destino. Por lo general, las apps para Android modernas usan una arquitectura de actividad única. En lugar de iniciar un nuevo Activity para cada pantalla, tu app aloja un solo Activity y usa el componente Navigation para intercambiar pantallas componibles dentro de esa actividad.

Para obtener información sobre cómo implementar una navegación moderna que priorice Compose, consulta la guía de la biblioteca de Navigation 3 de Jetpack Compose.

Cómo iniciar una actividad desde otra

Es posible que una actividad necesite iniciar otra actividad en algún momento. Esta necesidad surge, por ejemplo, cuando una app necesita pasar de la pantalla actual a una nueva.

En función de si la actividad desea recuperar el resultado de la nueva actividad que está a punto de comenzar, se puede iniciar la nueva actividad utilizando el método startActivity o el método startActivityForResult. En cualquier caso, debes pasar un objeto Intent.

El objeto Intent especifica la actividad exacta que quieres iniciar o describe el tipo de acción que quieres realizar. El sistema selecciona la actividad adecuada para ti, que incluso puede ser de otra aplicación. Un objeto Intent también puede contener pequeñas cantidades de datos que utilizará la actividad que se inicie. Para obtener más información sobre la clase Intent, consulta Intents y filtros de intents.

startActivity

Si la actividad recién iniciada no necesita mostrar un resultado, la actividad actual puede iniciarla llamando al método startActivity.

Cuando trabajes en tu propia aplicación, con frecuencia necesitarás iniciar una actividad conocida. Por ejemplo, el siguiente fragmento de código muestra cómo lanzar una actividad llamada SignInActivity.

val context = LocalContext.current

Button(onClick = {
    val intent = Intent(context, SignInActivity::class.java)
    context.startActivity(intent)
}) {
    Text("Sign In")
}

Cómo iniciar actividades externas

Si bien Navigation controla la navegación interna de la app, tu Activity deberá iniciar otras actividades de vez en cuando. Esto suele suceder cuando quieres aprovechar una aplicación externa para realizar una acción específica, como abrir un navegador web, enviar un correo electrónico o tomar una foto.

Para lograrlo, usa un objeto Intent para describir el tipo de acción que quieres realizar, y el sistema iniciará la actividad adecuada desde otra aplicación.

Por ejemplo, si quieres permitir que el usuario envíe un mensaje de correo electrónico, puedes crear el siguiente intent:

val intent = Intent(Intent.ACTION_SEND).apply {
    putExtra(Intent.EXTRA_EMAIL, recipientArray)
}
startActivity(intent)

Si necesitas iniciar una actividad externa y obtener un resultado (por ejemplo, pedirle a la app de la cámara que tome una foto y muestre la imagen), usa las APIs de resultados modernas Activity en lugar de la devolución de llamada startActivityForResult obsoleta.

Cómo coordinar actividades

Cuando una actividad inicia otra, ambas experimentan transiciones en su ciclo de vida. La primera actividad deja de funcionar y entra en el estado Paused o Stopped, mientras se crea la otra actividad. Si esas actividades comparten datos guardados en el disco o en alguna otra parte, es importante que entiendas que no se detiene la primera actividad por completo antes de que se cree la segunda. Más bien, el proceso de iniciar la segunda se superpone con el proceso de detener la primera.

El orden de las devoluciones de llamada del ciclo de vida está bien definido, en especial cuando las dos actividades están en el mismo proceso (en otras palabras, la misma app) y una inicia la otra. Aquí te mostramos el orden de las operaciones que ocurren cuando la actividad A inicia la actividad B:

  1. Se ejecuta el método onPause de la actividad A.
  2. Los métodos onCreate, onStart y onResume de la actividad B se ejecutan en secuencia. Ahora la actividad B tiene la atención del usuario.
  3. Si la actividad A ya no está visible en la pantalla, se ejecuta su método onStop.

Esta secuencia de devoluciones de llamada del ciclo de vida te permite administrar la transición de información de una actividad a otra.

Recursos adicionales

Para obtener más información sobre el ciclo de vida de la actividad, consulta los siguientes recursos adicionales:

Mira contenido