android– 由AlarmManager重新创建的服务

android– 由AlarmManager重新创建的服务,第1张

概述我有一个相当标准的服务,我希望使用警报触发.这是服务的启动部分:classMyServiceextendsService{privateContextcontext;privateAlarmManageralarmManager=null;privatefinalStringstartReason="com.stuff.myreason";privatefinalintREAS

我有一个相当标准的服务,我希望使用警报触发.这是服务的启动部分:

class MyService extends Service {    private Context context;    private AlarmManager  alarmManager = null;    private final String startReason = "com.stuff.myreason";    private final int REASON_NO_INTENT = 0;    private final int REASON_ALARM     = 1;    private final int REASON_X         = 2; // and so on.    @OverrIDe    voID onCreate() {        super.onCreate();        context = getApplicationContext();        alarmManager = (AlarmManager)getSystemService(ALARM_SERVICE);        // do onCreate stuff    }    @OverrIDe    int onStartCommand (Intent intent, int flags, int startID) {        int reason = REASON_NO_INTENT;        if (intent != null) {            reason = intent.getExtra(startReason, REASON_NO_INTENT);        }        switch(reason) {            // handle the different reasons we may have been "started"        }        return START_STICKY;    }}

当我使用来自活动的context.startService触发它时,它会正常启动.特别是,如果它已经在运行,它不会(重新)从头开始,而只是通过onStartCommand()进入现有的实例化.这是预期的行为.但是,当我使用AlarmManager触发它时:

Intent intent = new Intent(context, MyService.class);intent.putExtra(purpoSEOfStartCode, REASON_ALARM);PendingIntent pi = PendingIntent.getService(context, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT);alarmManager.set(AlarmManager.RTC_WAKEUP, /* A time in the future */, pi);

当警报到期时,它似乎从头开始重新启动服务:它启动一个新的实例化,调用onCreate()然后调用onStartCommand(),而不是在已经运行的实例化中调用onStartCommand().

我已经尝试将PendingIntent标志更改为FLAG_ONE_SHOT并将上下文替换为MyService.this而没有任何改进.

我对此感到很困惑 – 任何人都可以解释这种行为并建议如何让它按预期行事吗?

编辑 – 导致解决方案的行动集合在我的答案中.

解决方法:

经过一番调查和工作,我发现了很多东西.完成所有这些后,这个问题看起来已经消失了:

>如果您在服务中覆盖onStart和onStartCommand(以允许使用旧设备)并且将super.onStartCommand放在后者中,它将调用onStart,这意味着您将获得两次所有意图!
>根据其他答案之一(及其评论),AlarmManager的设计和指定是为了提供广播意图,而不是其他类型.然而,在实践中,它并不挑剔,似乎尊重其他形式.我认为这是解决问题的关键之一.
>如果服务与其他活动处于相同的过程中,则服务有时似乎“刚刚重新启动”.这可能是此问题中提到的问题的实际原因.见Android service onCreate is called multiple times without calling onDestroy.
>当仅使用意图与服务进行通信而不是绑定和使用Messenger或绑定和访问方法时,事情似乎更稳定.虽然这些都是正确的,但它们管理起来相当复杂(尽管您可以使用这种方法:What is the preferred way to call an Android Activity back from a Service thread和Using the Android Application class to persist data).虽然我完全理解androID文档不同意我,但在我的观察中,转向广播意图只是沟通似乎很关键.如果你采用单独的过程方法,你无论如何都必须这样做.
>在声明和解决课程方面保持一致是值得的.这有点乱,但是,因为有时似乎需要使用全名(“com.company.superapp.CLeverService”)而不是short(“CLeverService”或“.CLeverService”).因此,总是使用全名更好.
>关于上下文浮动的经验法则(“使用getApplicationContext”)实际上并不是正确的方法.见When to call activity context OR application context?;本质上使用它,除非你真的需要使用更广泛的上下文,并很好地管理你的变量.
>如果垃圾收集器在不再存在的Activity,Service,Thread,AsyncTask等中创建,它可能会清除仍在使用的内容.如果应用程序基于服务,那么制作一个类的副本可能是明智的,这样它们就不会在以后被清除.
>启动服务的比通常建议的更简洁的方法是为服务提供一个intentFilter,其全名作为 *** 作.然后,您可以使用类名作为字符串创建启动它的意图.这意味着您不必担心上下文.见Issue in Calling Service.

总结

以上是内存溢出为你收集整理的android – 由AlarmManager重新创建的服务全部内容,希望文章能够帮你解决android – 由AlarmManager重新创建的服务所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址:https://54852.com/web/1102479.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2022-05-28
下一篇2022-05-28

发表评论

登录后才能评论

评论列表(0条)

    保存