Cómo migrar servicios en primer plano a tareas de transferencia de datos iniciados por el usuario

对于应用何时可以使用前台服务,Android 14 制定了严格的规则

此外,在 Android 14 中,我们还引入了一个新 API,用于指定作业必须是用户发起的数据传输作业。此 API 适用于需要由用户发起的持续时间较长的数据传输,例如从远程服务器下载文件。这些类型的任务应使用由用户发起的数据传输作业。

由用户发起的数据传输作业由用户启动。这些作业需要一个通知,会立即启动,并且可能在系统条件允许的情况下长时间运行。您可以同时运行多个由用户发起的数据传输作业。

必须在应用对用户可见的情况下(或在某个允许的条件下)安排由用户发起的作业。满足所有限制条件后,操作系统可以执行由用户发起的作业,具体取决于系统运行状况限制。系统还可以根据提供的估算载荷大小来确定作业的执行时长。

Permiso para tareas de transferencia de datos que inicia el usuario

Las tareas de transferencia de datos que inicia el usuario requieren un permiso nuevo para ejecutarse: RUN_USER_INITIATED_JOBS. El sistema otorga este permiso automáticamente. El sistema arroja una SecurityException si no declaras el permiso en el manifiesto de la app.

Proceso para programar tareas de transferencia de datos que inicia el usuario

Para ejecutar una tarea que inicia el usuario, haz lo siguiente:

  1. Si es la primera vez que declaras una API con JobScheduler, declara el JobService y los permisos asociados en tu manifiesto. Además, define una subclase concreta de JobService para tu transferencia de datos:

    <service android:name="com.example.app.CustomTransferService"
            android:permission="android.permission.BIND_JOB_SERVICE"
            android:exported="false">
            ...
    </service>
    
    class CustomTransferService : JobService() {
      ...
    }
    
  2. Declara el permiso RUN_USER_INITIATED_JOBS en tu manifiesto:

    <manifest ...>
        <uses-permission android:name="android.permission.RUN_USER_INITIATED_JOBS" />
        <application ...>
            ...
        </application>
    </manifest>
    
  3. Llama al nuevo método setUserInitiated() cuando compiles un objeto JobInfo. También se recomienda que ofrezcas una estimación del tamaño de la carga útil mediante una llamada a setEstimatedNetworkBytes() mientras creas el trabajo:

    val networkRequestBuilder = NetworkRequest.Builder()
            .addCapability(NET_CAPABILITY_INTERNET)
            .addCapability(NET_CAPABILITY_NOT_METERED)
            // Add or remove capabilities based on your requirements
            .build()
    
    val jobInfo = JobInfo.Builder()
            // ...
            .setUserInitiated(true)
            .setRequiredNetwork(networkRequestBuilder.build())
            .setEstimatedNetworkBytes(1024 * 1024 * 1024)
            // ...
            .build()
    
  4. Programa la tarea antes de que comience la transferencia, mientras la aplicación está visible o en la lista de condiciones permitidas:

    val jobScheduler: JobScheduler =
        context.getSystemService(Context.JOB_SCHEDULER_SERVICE) as JobScheduler
    jobScheduler.schedule(jobInfo)
    
  5. Cuando se ejecute la tarea, asegúrate de llamar a setNotification() en el objeto JobService. Este valor se usa para informarle al usuario que la tarea está en ejecución, tanto en el Administrador de tareas como en el área de notificaciones de la barra de estado:

    class CustomTransferService : JobService() {
      override fun onStartJob(params: JobParameters?): Boolean {
          val notification = Notification.Builder(applicationContext, NOTIFICATION_CHANNEL_ID)
                  .setContentTitle("My user-initiated data transfer job")
                  .setSmallIcon(android.R.mipmap.myicon)
                  .setContentText("Job is running")
                  .build()
    
          setNotification(params, notification.id, notification,
                  JobService.JOB_END_NOTIFICATION_POLICY_DETACH)
          // Do the job execution.
      }
    }
    
  6. Actualiza la notificación de forma periódica para mantener informado al usuario sobre el estado y el progreso de la tarea. Si no puedes determinar el tamaño de transferencia antes de programar la tarea o necesitas actualizar el tamaño de transferencia estimado, usa la API nueva, updateEstimatedNetworkBytes(), para actualizar el tamaño de transferencia una vez que se haga conocido.

  7. Cuando finalice la ejecución, llama a jobFinished() para indicarle al sistema que se completó la tarea o que se debe reprogramar.

Cómo detener las tareas de transferencia de datos que inicia el usuario

El usuario y el sistema pueden detener tareas de transferencia que inicia el usuario.

Por el usuario, desde el Administrador de tareas

用户可以停止显示在任务管理器中的用户发起的传输作业。

在用户按 Stop 时,系统会执行以下操作:

  • 立即终止应用的进程,包括正在运行的所有其他作业或前台服务。
  • 不针对任何正在运行的作业调用 onStopJob()
  • 阻止重新调度用户可见的作业。

因此,建议在发布的作业通知中提供控件,以便顺利停止和重新调度作业。

请注意,在特殊情况下,Stop 按钮不会显示在任务管理器中的作业旁边,或者该作业根本不会显示在任务管理器中。

Por el sistema

A diferencia de las tareas normales, las de transferencia de datos que inicia el usuario no se ven afectadas por las cuotas de los buckets de App Standby. Sin embargo, el sistema detiene la tarea si se produce alguna de las siguientes condiciones:

  • Ya no se cumple una restricción que define el desarrollador.
  • El sistema determina que la tarea se ejecutó más tiempo del necesario para completar la tarea de transferencia de datos.
  • El sistema debe priorizar el estado del sistema y detener las tareas debido al aumento del estado térmico.
  • El proceso de la app finaliza debido a la poca memoria del dispositivo.

Cuando el sistema detiene la tarea (pero no por la poca memoria del dispositivo), el sistema llama a onStopJob() y vuelve a intentar la tarea en un momento que considera óptimo. Verifica que tu app pueda conservar el estado de transferencia de datos, incluso si no se llama a onStopJob(), y que tu app pueda restablecer este estado cuando se vuelva a llamar a onStartJob().

Condiciones permitidas para programar tareas de transferencia de datos que inicia el usuario

Las apps solo pueden comenzar una tarea de transferencia de datos que inicie el usuario si estas están en la ventana visible o si se cumplen ciertas condiciones. Para determinar cuándo se puede programar una tarea de transferencia de datos que inicia el usuario, el sistema aplica la misma lista de condiciones que permiten que las apps comiencen una actividad en segundo plano en casos especiales. En particular, esta lista de condiciones no es la misma que el conjunto de exenciones para las restricciones del servicio en primer plano que se inician en segundo plano.

Las excepciones a la declaración anterior son las siguientes:

  • Si una app puede iniciar actividades en segundo plano, también puede iniciar tareas de transferencia de datos que inicie el usuario en segundo plano.
  • Si una app tiene una actividad en la pila de actividades de una tarea existente en la pantalla Recientes, eso solo no permite que se ejecute una tarea de transferencia de datos que inicia el usuario.

Si la tarea está programada en algún otro momento que no esté en la lista de condiciones permitidas, la tarea falla y muestra un código de error RESULT_FAILURE.

Restricciones permitidas para las tareas de transferencia de datos que inicia el usuario

Para admitir las tareas que se ejecutan en puntos óptimos, Android ofrece la capacidad de asignar restricciones a cada tipo de tarea. Estas restricciones ya están disponibles a partir de Android 13.

Nota: En la siguiente tabla, solo se comparan las restricciones que varían entre cada tipo de tarea. Consulta la página del desarrollador de JobScheduler o las restricciones de tareas para conocer todas las restricciones.

En la siguiente tabla, se muestran los diferentes tipos de tareas que admiten una restricción de tarea determinada, así como el conjunto de restricciones de tareas que admite WorkManager. Usa la barra de búsqueda antes de la tabla para filtrarla por el nombre de un método de restricción de tarea.

Estas son las restricciones permitidas para las tareas de transferencia de datos que inicia el usuario:

  • setBackoffCriteria(JobInfo.BACKOFF_POLICY_EXPONENTIAL)
  • setClipData()
  • setEstimatedNetworkBytes()
  • setMinimumNetworkChunkBytes()
  • setPersisted()
  • setNamespace()
  • setRequiredNetwork()
  • setRequiredNetworkType()
  • setRequiresBatteryNotLow()
  • setRequiresCharging()
  • setRequiresStorageNotLow()

Pruebas

下面列出了有关如何手动测试应用作业的一些步骤:

  • 如需获取作业 ID,请获取在构建作业时定义的值。
  • 如需立即运行作业或重试已停止的作业,请在终端窗口中运行以下命令:

    adb shell cmd jobscheduler run -f APP_PACKAGE_NAME JOB_ID
    
  • 如需模拟系统强行停止作业(由于系统运行状况或超出配额条件),请在终端窗口中运行以下命令:

    adb shell cmd jobscheduler timeout TEST_APP_PACKAGE TEST_JOB_ID