Iniciar servicio en Android

115

Quiero llamar a un servicio cuando comienza una determinada actividad. Entonces, aquí está la clase de servicio:

public class UpdaterServiceManager extends Service {

    private final int UPDATE_INTERVAL = 60 * 1000;
    private Timer timer = new Timer();
    private static final int NOTIFICATION_EX = 1;
    private NotificationManager notificationManager;

    public UpdaterServiceManager() {}

    @Override
    public IBinder onBind(Intent intent) {
        // TODO Auto-generated method stub
        return null;
    }

    @Override
    public void onCreate() {
        // Code to execute when the service is first created
    }

    @Override
    public void onDestroy() {
        if (timer != null) {
            timer.cancel();
        }
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startid) {
        notificationManager = (NotificationManager) 
                getSystemService(Context.NOTIFICATION_SERVICE);
        int icon = android.R.drawable.stat_notify_sync;
        CharSequence tickerText = "Hello";
        long when = System.currentTimeMillis();
        Notification notification = new Notification(icon, tickerText, when);
        Context context = getApplicationContext();
        CharSequence contentTitle = "My notification";
        CharSequence contentText = "Hello World!";
        Intent notificationIntent = new Intent(this, Main.class);
        PendingIntent contentIntent = PendingIntent.getActivity(this, 0,
                notificationIntent, 0);
        notification.setLatestEventInfo(context, contentTitle, contentText,
                contentIntent);
        notificationManager.notify(NOTIFICATION_EX, notification);
        Toast.makeText(this, "Started!", Toast.LENGTH_LONG);
        timer.scheduleAtFixedRate(new TimerTask() {

            @Override
            public void run() {
                // Check if there are updates here and notify if true
            }
        }, 0, UPDATE_INTERVAL);
        return START_STICKY;
    }

    private void stopService() {
        if (timer != null) timer.cancel();
    }
}

Y así es como lo llamo:

Intent serviceIntent = new Intent();
serviceIntent.setAction("cidadaos.cidade.data.UpdaterServiceManager");
startService(serviceIntent);

El problema es que no pasa nada. El bloque de código anterior se llama al final de la actividad onCreate. Ya depuré y no se lanza ninguna excepción.

¿Alguna idea?

Miguel Ribeiro
fuente
1
Cuidado con los temporizadores: AFAIK cuando su servicio se cierra para liberar recursos, este temporizador no se reiniciará cuando se reinicie el servicio. Tiene razón START_STICKY, reiniciará el servicio, pero solo se llamará a onCreate y no se reiniciará la var del temporizador. Puede jugar con START_REDELIVER_INTENTel servicio de alarma o el programador de trabajos API 21 para solucionar este problema.
Georg
En caso de que lo haya olvidado, asegúrese de haber registrado el servicio en el manifiesto de Android usando <service android:name="your.package.name.here.ServiceClass" />la etiqueta de la aplicación.
Japheth Ongeri - inkalimeva

Respuestas:

278

Probablemente no tenga el servicio en su manifiesto, o no tenga un servicio <intent-filter>que coincida con su acción. Examinar LogCat (a través de adb logcat, DDMS o la perspectiva de DDMS en Eclipse) debería mostrar algunas advertencias que pueden ayudar.

Lo más probable es que deba iniciar el servicio a través de:

startService(new Intent(this, UpdaterServiceManager.class));
CommonsWare
fuente
1
¿Cómo se puede depurar? nunca llamé a mi servicio, mi depuración no muestra nada
entrega el
Agregue un montón de etiquetas Log.e en todas partes: antes de iniciar el servicio, el resultado de la intención del servicio, dentro de la clase de servicio a la que viajaría (onCreate, onDestroy, todos y cada uno de los métodos).
Zoe
funciona para mis aplicaciones en android sdk 26+ pero la dosis no en android sdk 25 o inferior. ¿Hay alguna solución?
Mahidul Islam
@MahidulIslam: le recomiendo que haga una pregunta separada de Stack Overflow, donde puede proporcionar un ejemplo mínimo reproducible que explique su problema y síntomas con mayor detalle.
CommonsWare
@CommonsWare ya hice una pregunta y esa es: - stackoverflow.com/questions/49232627/…
Mahidul Islam
81
startService(new Intent(this, MyService.class));

Simplemente escribir esta línea no fue suficiente para mí. El servicio aún no funcionaba. Todo había funcionado solo después de registrar el servicio en el manifiesto

<application
    android:icon="@drawable/ic_launcher"
    android:label="@string/app_name" >

    ...

    <service
        android:name=".MyService"
        android:label="My Service" >
    </service>
</application>
Vitalii Korsakov
fuente
1
El mejor ejemplo para aprender todo sobre los servicios en Android coderzpassion.com/implement-service-android y perdón por llegar tarde
Jagjit Singh
55

Código Java para iniciar el servicio :

Iniciar el servicio desde la actividad :

startService(new Intent(MyActivity.this, MyService.class));

Iniciar servicio desde Fragmento :

getActivity().startService(new Intent(getActivity(), MyService.class));

MyService.java :

import android.app.Service;
import android.content.Intent;
import android.os.Handler;
import android.os.IBinder;
import android.util.Log;

public class MyService extends Service {

    private static String TAG = "MyService";
    private Handler handler;
    private Runnable runnable;
    private final int runTime = 5000;

    @Override
    public void onCreate() {
        super.onCreate();
        Log.i(TAG, "onCreate");

        handler = new Handler();
        runnable = new Runnable() {
            @Override
            public void run() {

                handler.postDelayed(runnable, runTime);
            }
        };
        handler.post(runnable);
    }

    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }

    @Override
    public void onDestroy() {
        if (handler != null) {
            handler.removeCallbacks(runnable);
        }
        super.onDestroy();
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        return START_STICKY;
    }

    @SuppressWarnings("deprecation")
    @Override
    public void onStart(Intent intent, int startId) {
        super.onStart(intent, startId);
        Log.i(TAG, "onStart");
    }

}

Defina este servicio en el archivo de manifiesto del proyecto:

Agregue la siguiente etiqueta en el archivo de manifiesto :

<service android:enabled="true" android:name="com.my.packagename.MyService" />

Hecho

Hiren Patel
fuente
7
¿Cuánto mejora el rendimiento cuando dejo actividades y servicios en el mismo paquete? Nunca escuché eso antes.
OneWorld
¿Quizás se referían al rendimiento en un sentido vago y vago, no a la velocidad de carrera?
Anubian Noob
3

Me gusta hacerlo mas dinámico

Class<?> serviceMonitor = MyService.class; 


private void startMyService() { context.startService(new Intent(context, serviceMonitor)); }
private void stopMyService()  { context.stopService(new Intent(context, serviceMonitor));  }

no te olvides del manifiesto

<service android:enabled="true" android:name=".MyService.class" />
Thiago
fuente
1
Intent serviceIntent = new Intent(this,YourActivity.class);

startService(serviceIntent);

agregar servicio en colector

<service android:enabled="true" android:name="YourActivity.class" />

para ejecutar el servicio en oreo y dispositivos más grandes, utilizar para el servicio terrestre y mostrar una notificación al usuario

o use el servicio de geofencing para actualizar la ubicación en la referencia de fondo http://stackoverflow.com/questions/tagged/google-play-services

Asif Mehmood
fuente