Android - Uruchom usługę podczas rozruchu

107

Ze wszystkiego, co widziałem na Stack Exchange i gdzie indziej, mam wszystko poprawnie skonfigurowane, aby uruchomić IntentService po uruchomieniu systemu operacyjnego Android. Niestety nie uruchamia się przy starcie i nie pojawiają się żadne błędy. Może eksperci pomogą ...

Oczywisty:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
  package="com.phx.batterylogger"
  android:versionCode="1"
  android:versionName="1.0"
  android:installLocation="internalOnly">

<uses-sdk android:minSdkVersion="8" />
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.BATTERY_STATS" />

<application android:icon="@drawable/icon" android:label="@string/app_name">
    <service android:name=".BatteryLogger"/>
    <receiver android:name=".StartupIntentReceiver">  
        <intent-filter>  
            <action android:name="android.intent.action.BOOT_COMPLETED" />  
        </intent-filter>  
    </receiver>
</application>

</manifest>

BroadcastReceiver for Startup:

package com.phx.batterylogger;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;

public class StartupIntentReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Intent serviceIntent = new Intent(context, BatteryLogger.class);
        context.startService(serviceIntent);
    }
}

AKTUALIZACJA : Wypróbowałem prawie wszystkie poniższe sugestie i dodałem rejestrowanie, takie jak Log.v("BatteryLogger", "Got to onReceive, about to start service");do programu obsługi onReceive w StartupIntentReceiver, i nic nie jest nigdy rejestrowane. Więc nawet nie trafia do BroadcastReceiver.

Myślę, że wdrażam pakiet APK i testuję poprawnie, po prostu uruchamiam debugowanie w Eclipse, a konsola informuje, że pomyślnie instaluje go na moim tablecie Xoom w \ BatteryLogger \ bin \ BatteryLogger.apk. Następnie, aby przetestować, ponownie uruchamiam tablet, a następnie przeglądam dzienniki w DDMS i sprawdzam Running Services w ustawieniach systemu operacyjnego. Czy to wszystko brzmi poprawnie, czy czegoś mi brakuje? Jakakolwiek pomoc jest mile widziana.

Gady
źródło
1
jaki masz problem, czy nie dostajesz żadnego interfejsu użytkownika?
Lalit Poptani,
1
Usługa po prostu nigdy się nie uruchamia, to jest problem.
Gady,
skąd wiesz, że Twoja usługa się nie rozpoczyna, czy wydrukowałeś dzienniki lub coś w tym rodzaju?
Lalit Poptani
1
Nie potrzebujesz dzienników, aby zobaczyć, że nie działa. System operacyjny Android ujawnia uruchomione usługi. Jednak rozsądnie byłoby skorzystać z rejestrowania, aby sprawdzić, czy występuje błąd. Zaryzykowałbym przypuszczenie, że dzieje się to przed context.startService (), jeśli wystąpi błąd.
Tony
1
Dodałem Log.v("BatteryLogger", "Got to onReceive, about to start service");do modułu obsługi onReceive i nigdy nie pojawia się w dziennikach. A więc słuchacz zawodzi (?)
Gady

Odpowiedzi:

302

Oto kompletny przykład aplikacji AutoStart

Plik AndroidManifest

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="pack.saltriver" android:versionCode="1" android:versionName="1.0">

    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />

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

        <receiver android:name=".autostart">
            <intent-filter>
                <action android:name="android.intent.action.BOOT_COMPLETED" />
            </intent-filter>
        </receiver>

        <activity android:name=".hello"></activity>
        <service android:enabled="true" android:name=".service" />
    </application>
</manifest>

autostart.java

public class autostart extends BroadcastReceiver 
{
    public void onReceive(Context context, Intent arg1) 
    {
        Intent intent = new Intent(context,service.class);
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            context.startForegroundService(intent);
        } else {
            context.startService(intent);
        }
        Log.i("Autostart", "started");
    }
}

service.java

public class service extends Service
{
    private static final String TAG = "MyService";
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }
    public void onDestroy() {
        Toast.makeText(this, "My Service Stopped", Toast.LENGTH_LONG).show();
        Log.d(TAG, "onDestroy");
    }

    @Override
    public void onStart(Intent intent, int startid)
    {
        Intent intents = new Intent(getBaseContext(),hello.class);
        intents.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        startActivity(intents);
        Toast.makeText(this, "My Service Started", Toast.LENGTH_LONG).show();
        Log.d(TAG, "onStart");
    }
}

hello.java - wyskakuje przy każdym uruchomieniu urządzenia po jednokrotnym uruchomieniu aplikacji.

public class hello extends Activity 
{   
    public void onCreate(Bundle savedInstanceState) 
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        Toast.makeText(getBaseContext(), "Hello........", Toast.LENGTH_LONG).show();
    }
}
Lalit Poptani
źródło
10
+1 za pełny przykład. Może powinienem zmienić moją usługę IntentService z onHandleIntent na zwykłą starą usługę
Gady,
6
To ostatecznie rozwiązało problem. Problem polegał na połączeniu mojej usługi jako IntentService zamiast zwykłej starej usługi i błędów w kodzie usługi. Dokładność tej odpowiedzi i zalogowanie się na Androida pomogło mi rozwiązać moje problemy.
Gady
3
+1 za szczegółowy przykład. Czy to również zadziałałoby bez aktywności?
balas
1
Funkcja onStart () jest przestarzała. Zamiast tego należy użyć onStartCommand ().
mmBs
1
@mmBs ta odpowiedź ma prawie 5 lat, więc jest to oczywiste
Lalit Poptani
3

Twoja usługa może być wyłączana przed zakończeniem, ponieważ urządzenie przechodzi w stan uśpienia po uruchomieniu. Najpierw musisz uzyskać blokadę budzenia. Na szczęście biblioteka pomocy technicznej daje nam do tego klasę :

public class SimpleWakefulReceiver extends WakefulBroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        // This is the Intent to deliver to our service.
        Intent service = new Intent(context, SimpleWakefulService.class);

        // Start the service, keeping the device awake while it is launching.
        Log.i("SimpleWakefulReceiver", "Starting service @ " + SystemClock.elapsedRealtime());
        startWakefulService(context, service);
    }
}

następnie w swojej usłudze pamiętaj o zwolnieniu blokady uśpienia:

    @Override
    protected void onHandleIntent(Intent intent) {
        // At this point SimpleWakefulReceiver is still holding a wake lock
        // for us.  We can do whatever we need to here and then tell it that
        // it can release the wakelock.

...
        Log.i("SimpleWakefulReceiver", "Completed service @ " + SystemClock.elapsedRealtime());
        SimpleWakefulReceiver.completeWakefulIntent(intent);
    }

Nie zapomnij dodać uprawnienia WAKE_LOCK:

<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
phreakhead
źródło
1
Mam ten sam problem. Możesz mi pomóc? Dzięki! stackoverflow.com/questions/35373525/starting-my-service
Ruchir Baronia
2

Następujące powinno działać. Zweryfikowałem. Być może twój problem jest gdzie indziej.

public class MyReceiver extends BroadcastReceiver{

    @Override
    public void onReceive(Context context, Intent intent) {
        Log.d("TAG", "MyReceiver");
        Intent serviceIntent = new Intent(context, Test1Service.class);
        context.startService(serviceIntent);
    }
}




public class Test1Service extends Service {
    /** Called when the activity is first created. */
    @Override
    public void onCreate() {
        super.onCreate();
        Log.d("TAG", "Service created.");
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        Log.d("TAG", "Service started.");
        return super.onStartCommand(intent, flags, startId);
    }

    @Override
    public void onStart(Intent intent, int startId) {
        super.onStart(intent, startId);
        Log.d("TAG", "Service started.");
    }
    @Override
    public IBinder onBind(Intent arg0) {
        return null;
    }
}




<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
      package="com.test"
      android:versionCode="1"
      android:versionName="1.0"
      android:installLocation="internalOnly">
    <uses-sdk android:minSdkVersion="8" />

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

    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.BATTERY_STATS" 
    />
<!--        <activity android:name=".MyActivity">
            <intent-filter>  
                <action android:name="android.intent.action.MAIN" /> 
                <category android:name="android.intent.category.LAUNCHER"></category> 
            </intent-filter>
       </activity> -->
        <service android:name=".Test1Service" 
                  android:label="@string/app_name"
                  >
        </service>
        <receiver android:name=".MyReceiver">  
            <intent-filter>  
                <action android:name="android.intent.action.BOOT_COMPLETED" /> 
            </intent-filter>  
        </receiver> 
    </application>
</manifest>
Vivek
źródło
Dzięki za to, ale nadal nie działa. Dzienniki nawet nie pokazują, że trafia do StartupIntentReceiver. Jakieś inne pomysły?
Gady,
1
Czy Twój kod działa w emulatorze? Czy możesz otrzymać zamiar w emulatorze?
Vivek,
Testowałem w emulatorze i otrzymuję błąd w DDMS, ale wygląda na to, że przynajmniej usługa próbuje się uruchomić, chociaż nie Log()ma tam żadnych moich instrukcji, a urządzenie emulatora nie pokazuje mojej usługi jako działającej w ustawieniach systemu operacyjnego. Oto błąd w DDMS: System.err - at com.phx.batterylogger$1.onReceive(BatteryLogger.java:43)Czy to oznacza, że ​​problem występuje w linii 43 mojej usługi BatteryLogger?
Gady
2

Znalazłem sposób, aby aplikacja działała dobrze po ponownym uruchomieniu urządzenia. Aby odnieść sukces, wykonaj poniższe czynności.

Plik AndroidManifest

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="pack.saltriver" android:versionCode="1" android:versionName="1.0">
<uses-permission 
android:name="android.permission.REQUEST_IGNORE_BATTERY_OPTIMIZATIONS"/>
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />

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

<activity android:name=".MainActivity">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>
    <receiver android:name=".UIBootReceiver" android:enabled="true" 
    android:exported="true">
        <intent-filter>
            <action android:name="android.intent.action.BOOT_COMPLETED" />
            <category android:name="android.intent.category.DEFAULT" />
        </intent-filter>
    </receiver>
     <service android:name=".class_Service" />
</application>

UIBootReceiver

public class UIBootReceiver extends BroadcastReceiver {

private static final String TAG = "UIBootReceiver";
@Override

    public void onReceive(Context context, Intent arg1)
    {
        Toast.makeText(context, "started", Toast.LENGTH_SHORT).show();
        Intent intent = new Intent(context,class_Service.class);
        context.startService(intent);
    }
  }

Jest to prośba o pozwolenie, aby nie trzeba było zarządzać oszczędzaniem baterii dla tej aplikacji, aby można było stabilnie działać w tle.

Zadeklaruj ten kod w onCreate () klasy MainActivity:

   Intent myIntent = new Intent();
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
        myIntent.setAction(Settings.ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS);
        myIntent.setData(Uri.parse("package:" + 
   DeviceMovingSpeed.this.getPackageName()));
    }
    startActivity(myIntent);
Huy TRAN
źródło
1

Wygląda bardzo podobnie do mojego, ale używam pełnej nazwy pakietu dla odbiorcy:

<receiver android:name=".StartupIntentReceiver">

Mam:

<receiver android:name="com.your.package.AutoStart"> 
ciscogambo
źródło
Tak, trochę było, aby rozwiązać problem, więc lepiej opublikuj przykład Hello World.
Lalit Poptani
1

Odniosłem sukces bez pełnego pakietu, czy wiesz, gdzie łańcuch połączeń jest przerywany? Jeśli debugujesz zLog() , w którym momencie to już nie działa?

Myślę, że może to być w Twojej IntentService, wszystko wygląda dobrze.

Phix
źródło
Dodałem Log.v("BatteryLogger", "Got to onReceive, about to start service");do modułu obsługi onReceive i nigdy nie pojawia się w dziennikach. A więc słuchacz zawodzi (?)
Gady
Co sprawia, że ​​odbiornik transmisji uruchamia się po uruchomieniu telefonu? Dzięki!
Ruchir Baronia
Chłopak. Od kilku lat nie tknąłem androida. Przepraszam @Ruchir
Phix
0

Aby ułatwić wyszukiwanie, jak wspomniano w komentarzach, nie jest to możliwe od 3.1 https://stackoverflow.com/a/19856367/6505257

MrKew
źródło
To nieprawda, jak stwierdził Think Twice Code Once w sekcji komentarzy tej samej odpowiedzi.
Ricardo A.
Bardziej szczegółowe wyjaśnienie na poparcie mojego argumentu: stackoverflow.com/questions/20441308/…
Ricardo A.
Minęło dużo czasu odkąd nic nie robiłem z androidem, ale masz rację, jest to możliwe. Żadne inne argumenty nie są potrzebne. (
Poprawię