Запустить сервис в Android

115

Я хочу позвонить в службу, когда начнется определенное действие. Итак, вот класс 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();
    }
}

И вот как я это называю:

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

Проблема в том, что ничего не происходит. Вышеупомянутый блок кода вызывается в конце действия onCreate. Я уже отлаживал, и никаких исключений не возникало.

Любая идея?

Мигель Рибейро
источник
1
Будьте осторожны с таймерами - AFAIK, когда ваша служба отключена для освобождения ресурсов, этот таймер не будет перезапущен при перезапуске службы. Вы правы START_STICKY, перезапустите службу, но тогда вызывается только onCreate, и переменная таймера не будет повторно инициализирована. START_REDELIVER_INTENTЧтобы исправить это, вы можете поиграть со службой тревог или планировщиком заданий API 21.
Георг
Если вы забыли, убедитесь, что вы зарегистрировали службу в манифесте Android, используя <service android:name="your.package.name.here.ServiceClass" />тег приложения.
Иафет Онгери - инкалимева

Ответы:

279

Возможно, у вас нет службы в вашем манифесте или в ней нет <intent-filter>соответствующего действия. Изучение LogCat (с помощью adb logcatDDMS или перспективы DDMS в Eclipse) должно выявить некоторые предупреждения, которые могут помочь.

Скорее всего, вам следует запустить службу через:

startService(new Intent(this, UpdaterServiceManager.class));
CommonsWare
источник
1
Как можно отладить? никогда не звонил в мою службу, мой отладчик ничего не показывает
Добавьте кучу тегов Log.e повсюду: перед запуском службы, результат намерения службы, внутри класса обслуживания, по которому он будет перемещаться (onCreate, onDestroy, любые и все методы).
Zoe
это работает для моих приложений на android sdk 26+, но не на android sdk 25 или ниже. есть какое решение?
Mahidul Islam
@MahidulIslam: я рекомендую вам задать отдельный вопрос о переполнении стека, где вы можете предоставить минимальный воспроизводимый пример, более подробно объясняющий вашу проблему и симптомы.
CommonsWare
@CommonsWare, я уже задавал вопрос: - stackoverflow.com/questions/49232627/…
Mahidul Islam
81
startService(new Intent(this, MyService.class));

Просто написать эту строку для меня было недостаточно. Сервис по-прежнему не работал. Все заработало только после регистрации сервиса в манифесте

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

    ...

    <service
        android:name=".MyService"
        android:label="My Service" >
    </service>
</application>
Виталий Корсаков
источник
1
Лучший пример, чтобы узнать все об услугах в Android coderzpassion.com/implement-service-android и извиниться за опоздание
Джагджит Сингх
55

Код Java для запуска службы :

Запустить службу из Activity :

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

Запустить службу из фрагмента :

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

}

Определите эту службу в файле манифеста проекта:

Добавьте ниже тег в файл манифеста :

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

Готово

Хирен Патель
источник
7
Насколько улучшается производительность, если я оставляю мероприятия и услуги в одном пакете? Никогда об этом не слышал.
OneWorld,
Возможно, они имели в виду производительность в очень расплывчатом смысле, а не о скорости бега?
Anubian Noob
3

Мне нравится делать его более динамичным

Class<?> serviceMonitor = MyService.class; 


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

не забывай манифест

<service android:enabled="true" android:name=".MyService.class" />
Тьяго
источник
1
Intent serviceIntent = new Intent(this,YourActivity.class);

startService(serviceIntent);

добавить услугу в манифист

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

для запуска службы на устройствах oreo и выше, используйте для наземного обслуживания и показывайте уведомление пользователю

или используйте службу геозон для обновления местоположения в фоновой справке http://stackoverflow.com/questions/tagged/google-play-services

Асиф Мехмуд
источник