Android Service stops working, when app isn't

2019-03-01 22:29发布

问题:

I've got a little problem: I recieve from a Service a speed value given by the LocationListener. But when i close the ui of my app then the locationlistener stops sending updates. Does anyone have an idea what to do? I need it to continue the updates, even if the app is not in use...

Here is my code:

public class BackroundService extends Service {


//initializing the Location Manager and Listener
LocationManager locationManager;
LocationListener locationListener;


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

@SuppressLint("MissingPermission")
@Override
public void onCreate() {

}

@SuppressLint("MissingPermission")
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
    //Instantiating the device manager an  listener
    locationManager = (LocationManager) getSystemService(LOCATION_SERVICE);
    locationListener = new MyLocationListener();
    locationManager.requestLocationUpdates("gps", 500, 0, locationListener);

    return START_STICKY;
}




}


class MyLocationListener extends Service implements LocationListener
{
public float speed;

public void onLocationChanged(final Location location)
{
    //when the location changed

    Log.d("Location", ""+location);
    Log.d("Float Location", ""+ location.getLongitude() + "     " + location.getLatitude());
    Log.d("Speed", "    "+location.getSpeed());

    //initializing the variable speed with the speed number given by the LocationListener
    speed = location.getSpeed();

    //creating a broadcast reciever
    Intent intent = new Intent("speed_update");
    intent.putExtra("speed", speed);
    //sending speed to main activity
    sendBroadcast(intent);

    //checking if speed is in walking range
    if (speed < 4.0 && speed > 0.4){
        Log.d("######Checker######", "++++++++++turn off+++++++++++");
        //Toast.makeText(MyLocationListener.this, "turn off", Toast.LENGTH_SHORT).show();
    }
}


public void onProviderDisabled(String provider)
{

}


public void onProviderEnabled(String provider)
{

}


public void onStatusChanged(String provider, int status, Bundle extras)
{

}

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

回答1:

From oreo there's major changes in android background policy, please refer this https://developer.android.com/about/versions/oreo/background for more details, in order to avoid your service from getting killed by android you have to run it as a foreground service

Step 1: Add permission to manifest file

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

heres more details on this permission https://developer.android.com/reference/android/Manifest.permission#FOREGROUND_SERVICE

Step 2: Create service, I'm sharing example code for you

public class DemoService extends Service {
    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        Log.d("SSB Log", "onStartCommand");

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            // For foreground service
            Intent notificationIntent = new Intent(this, DemoService.class);
            PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, notificationIntent, 0);

            // Creating channel for notification
            String id = DemoService.class.getSimpleName();
            String name = DemoService.class.getSimpleName();
            NotificationChannel notificationChannel = new NotificationChannel(id,
                    name, NotificationManager.IMPORTANCE_NONE);
            NotificationManager service = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
            service.createNotificationChannel(notificationChannel);

            // Foreground notification
            Notification notification = new Notification.Builder(this, id)
                    .setContentTitle(getText(R.string.app_name))
                    .setContentText("Show service running reason to user")
                    .setSmallIcon(R.mipmap.ic_launcher)
                    .setContentIntent(pendingIntent)
                    .setTicker("Ticker text")
                    .build();

            startForeground(9, notification);
        }
        // Service logic here

        return Service.START_NOT_STICKY;
    }
}

Step 3: Declare service in manifest

<service android:name=".DemoService" />

Step 4: Start service from activity

Intent startDemoService = new Intent(this,DemoService.class);
    startService(startDemoService);