【问题标题】:service automatically stops after several minutes几分钟后服务自动停止
【发布时间】:2018-04-15 20:17:24
【问题描述】:

我正在创建一个服务,该服务应该在活动处于后台以及整个应用程序被销毁时工作。

我每隔 1 分钟调用一次服务中的位置坐标。

但是,当我尝试这样做时,服务会在 12-15 分钟后自动关闭。

我希望服务能够无休止地工作,直到并且除非它因用户交互活动的完成而被破坏。

我的服务类如下:

public class SensorService extends Service {
    public int counter=0; 
    public static final int NINTY_SECONDS = 90000; 
    public static Boolean isRunning = false;
    public LocationManager mLocationManager;
    public Get_Coordinates mLocationListener;
    public GetSharedPreference sharedPreference;
    @Override
    public void onCreate() {
        mLocationManager = (LocationManager) getSystemService(Context.LOCATION_SERVICE);
        mLocationListener = new Get_Coordinates(getApplicationContext());
        sharedPreference = new GetSharedPreference(getApplicationContext());
        startTimer();
        super.onCreate();
    }
    public SensorService(Context applicationContext) {
        super();
        Log.i("HERE", "here I am!");
    }

    public SensorService() {
    }
    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        super.onStartCommand(intent, flags, startId);
        return START_STICKY;
    }
    @Override
    public void onDestroy() {
        super.onDestroy();
        Log.i("EXIT", "ondestroy!");
        Intent broadcastIntent = new Intent("com.android.startBgService");
        broadcastIntent.putExtra("abc","abcd");
        sendBroadcast(broadcastIntent);
        stoptimertask();
    }

    private Timer timer;
    private TimerTask timerTask;
    long oldTime=0;
    public void startTimer() {
        //set a new Timer
        timer = new Timer();

        //initialize the TimerTask's job
        initializeTimerTask();

        //schedule the timer, to wake up every 1 second
        timer.schedule(timerTask, NINTY_SECONDS, NINTY_SECONDS); //
    }

    /**
     * it sets the timer to print the counter every x seconds
     */
    public void initializeTimerTask() {
        timerTask = new TimerTask() {
            public void run() {
               // Log.i("in Etro Sendor", "in timer ++++  "+ (counter++));
                if (Check_Internet_Con.isConnectingToInternet(getApplicationContext())) {

                    if (!isRunning) {
                        startListening();
                    }
                    try {
                        if (sharedPreference.getActiveUserId() > 0) {
                            mLocationListener.getLocation();
                            mLocationListener.insertCoordinatesInSqlite();
                        }
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            }
        };
    }

    /**
     * not needed
     */
    public void stoptimertask() {
        //stop the timer, if it's not already null
        if (timer != null) {
            timer.cancel();
            timer = null;
        }
    }

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }
    public void startListening() {
        if (ContextCompat.checkSelfPermission(this, Manifest.permission.ACCESS_FINE_LOCATION) == PackageManager.PERMISSION_GRANTED
                || ContextCompat.checkSelfPermission(this, Manifest.permission.ACCESS_COARSE_LOCATION) == PackageManager.PERMISSION_GRANTED) {
            if (mLocationManager.getAllProviders().contains(LocationManager.NETWORK_PROVIDER))
                mLocationManager.requestLocationUpdates(LocationManager.NETWORK_PROVIDER, 0, 0, mLocationListener, Looper.getMainLooper());

            if (mLocationManager.getAllProviders().contains(LocationManager.GPS_PROVIDER))
                mLocationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, 0, 0, mLocationListener,Looper.getMainLooper());
        }
        isRunning = true;
    }
}

这是我的清单

<service
           android:name="com.lunetta.etro.e_tro.SensorService"
           android:enabled="true"></service>
       <service
           android:name="com.lunetta.etro.e_tro.SecondService"
           android:enabled="true" >
       </service>

       <receiver
           android:name=".SensorRestarterBroadcastReceiver"
           android:enabled="true"
           android:exported="true"
           android:label="RestartServiceWhenStopped">
           <intent-filter>
               <action android:name="com.android.startBgService" />
               <action android:name="android.intent.action.BOOT_COMPLETED" />
           </intent-filter>
       </receiver>

这里是 SensorRestarterBroadcastReceiver 类

public class SensorRestarterBroadcastReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Log.i(SensorRestarterBroadcastReceiver.class.getSimpleName(), "Service Stops! Oooooooooooooppppssssss!!!!");
        context.startService(new Intent(context, SensorService.class));
        
    }
}

【问题讨论】:

标签: android broadcastreceiver android-service timertask


【解决方案1】:

https://developer.android.com/guide/components/bound-services.html

“绑定服务通常仅在它为另一个应用程序组件提供服务时才存在,并且不会无限期地在后台运行。”

要使其无限期运行,您需要将服务绑定到无限期存在的 UI 组件。您可以使用 Android 通知。这是前台服务。

https://developer.android.com/guide/components/services.html#Foreground

【讨论】:

    【解决方案2】:

    是的,它会停止工作,请查看 Google 发布的新文档,它完全改变了位置对电池性能的影响,是的,它对性能有很大的影响,我已经完全改变了我的具体任务.也停止使用其他声称它们工作错误的存储库

    通过谷歌检查新更新的文档

    https://developer.android.com/training/building-location.html

    我实现您的相同要求的最佳方式是并且可以在大约 60k 台设备中运行,并且可以完美地工作是使用 JobService 的 Android API 版本,使用 23 及更高版本和更低版本的后台服务。已相应地使用 LocationProvider API 和 Location provider Client API。

    根据我的个人经验,我会这么说 旧式的代码会在几个小时内耗尽设备电池,现在我的代码几乎没有任何影响,它仅消耗 15% 的过夜总使用量。这是消费的巨大变化。

    【讨论】:

    • 那么我该如何实现你的方式,如果有任何步骤可以指导我,这将非常有帮助。
    • 正如我告诉你的,你需要非常巧妙和熟练地实现相同的代码,你需要使用 JobService 表格 23 及更高版本,对于年龄较大的你需要使用和导出服务