Uruchom usługę w systemie Android

115

Chcę zadzwonić do serwisu, gdy rozpocznie się określona czynność. Oto klasa Service:

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();
    }
}

A oto jak to nazywam:

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

Problem w tym, że nic się nie dzieje. Powyższy blok kodu jest wywoływany na końcu działania onCreate. Mam już debugowanie i nie został zgłoszony żaden wyjątek.

Dowolny pomysł?

Miguel Ribeiro
źródło
1
Ostrożnie z licznikami czasu - AFAIK, gdy usługa jest wyłączona, aby zwolnić zasoby, ten licznik czasu nie zostanie ponownie uruchomiony po ponownym uruchomieniu usługi. Masz rację START_STICKY, zrestartujesz usługę, ale wtedy wywoływana jest tylko metoda onCreate, a zmienna timera nie zostanie ponownie zainicjowana. Możesz bawić się START_REDELIVER_INTENTusługą Alarm lub Harmonogramem zadań API 21, aby to naprawić.
Georg
Jeśli zapomniałeś, upewnij się, że zarejestrowałeś usługę w manifeście systemu Android przy użyciu <service android:name="your.package.name.here.ServiceClass" />tagu aplikacji.
Japheth Ongeri - inkalimeva

Odpowiedzi:

278

Prawdopodobnie nie masz usługi w swoim manifeście lub nie ma ona odpowiednika <intent-filter>odpowiadającego Twojemu działaniu. Analiza LogCat (via adb logcat, DDMS lub perspektywa DDMS w Eclipse) powinna ujawnić pewne ostrzeżenia, które mogą pomóc.

Bardziej prawdopodobne jest, że powinieneś uruchomić usługę przez:

startService(new Intent(this, UpdaterServiceManager.class));
CommonsWare
źródło
1
Jak możesz debugować? nigdy nie dzwonił do mojej usługi, mój debugowanie nic nie pokazuje
Dodaj gówno tagów Log.e wszędzie: przed uruchomieniem usługi, wynik zamiaru usługi, wewnątrz klasy usługi, do której ma podróżować (onCreate, onDestroy, wszystkie metody).
Zoe,
działa dla moich aplikacji na Androidzie SDK 26+, ale nie na Androidzie SDK 25 lub niższym. jest jakieś rozwiązanie?
Mahidul Islam
@MahidulIslam: Zalecam zadanie osobnego pytania o przepełnienie stosu, w którym możesz podać minimalny powtarzalny przykład, wyjaśniający bardziej szczegółowo problem i objawy.
CommonsWare,
@CommonsWare Już zadałem pytanie, a to: - stackoverflow.com/questions/49232627/ ...
Mahidul Islam
81
startService(new Intent(this, MyService.class));

Samo napisanie tego wiersza nie było dla mnie wystarczające. Usługa nadal nie działała. Wszystko zadziałało dopiero po zarejestrowaniu usługi w manifeście

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

    ...

    <service
        android:name=".MyService"
        android:label="My Service" >
    </service>
</application>
Vitalii Korsakov
źródło
1
Najlepszy przykład, aby dowiedzieć się wszystkiego o usługach w systemie Android coderzpassion.com/implement-service-android i przepraszam za spóźnienie
Jagjit Singh
55

Kod Java do uruchomienia usługi :

Uruchom usługę od Aktywności :

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

Uruchom usługę od fragmentu :

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");
    }

}

Zdefiniuj tę usługę w pliku manifestu projektu:

Dodaj poniższy tag w pliku manifestu :

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

Gotowe

Hiren Patel
źródło
7
Jak bardzo poprawia to wydajność, gdy zostawiam działania i usługi w tym samym pakiecie? Nigdy wcześniej tego nie słyszałem.
OneWorld
Może chodziło o wydajność w bardzo niejasnym, luźnym sensie, a nie o szybkości biegu?
Anubian Noob
3

Lubię, aby był bardziej dynamiczny

Class<?> serviceMonitor = MyService.class; 


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

nie zapominajcie o Manifeście

<service android:enabled="true" android:name=".MyService.class" />
Thiago
źródło
1
Intent serviceIntent = new Intent(this,YourActivity.class);

startService(serviceIntent);

dodaj usługę w pliku manifestu

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

do obsługi usługi na urządzeniach oreo i większych używaj do obsługi naziemnej i wyświetlaj powiadomienie użytkownikowi

lub użyj usługi geofencing do aktualizacji lokalizacji w tle http://stackoverflow.com/questions/tagged/google-play-services

Asif Mehmood
źródło