13

我有一个更新 SharedPreferences 中的字符串的活动。

SharedPreferences settings = PreferenceManager.getDefaultSharedPreferences(this);
SharedPreferences.Editor editor = settings.edit();
editor.putString("username", username);
editor.commit();

然后我启动一个服务:

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

该服务创建一个对扩展 BroadcastReceiver 的 Alarm 的引用:

Alarm alarm = null;
public void onCreate() {
    alarm = new Alarm();
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
    alarm.SetAlarm(this);
}

在 SetAlarm 中,我进行了所有基本设置(此时,“用户名”仍然正确。我检查了):

public void SetAlarm(Context context) {
    AlarmManager am=(AlarmManager)context.getSystemService(Context.ALARM_SERVICE);
    PendingIntent pi = PendingIntent.getBroadcast(context, 0, i, 0);
    am.setRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP, SystemClock.elapsedRealtime(), 1000 * 60 * interval, pi);
}

然后我停止该服务,然后再次启动它(使用 SetAlarm)。

public void CancelAlarm(Context context) {
   Intent intent = new Intent(context, Alarm.class);
   PendingIntent sender = PendingIntent.getBroadcast(context, 0, intent, 0);
   AlarmManager alarmManager = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
   alarmManager.cancel(sender);
}

问题出在 onReceive 中。第一次“用户名”字段是正确的。第二次,如果用户名在服务停止和启动之间更新,但是,它返回第一个值。该值似乎没有更新...

public void onReceive(Context context, Intent intent) {   
    SharedPreferences settings = PreferenceManager.getDefaultSharedPreferences(context);
    Log.e("hi", settings.getString("username", ""));
}
4

2 回答 2

67

我遇到了同样的问题,经过几个小时的努力解决它,我终于找到了导致它的问题。在您的 AndroidManifest 中,您可能有类似的内容:

<receiver android:name="AlarmReceiver" android:process=":remote" />

最后一个属性 (process:remote) 导致接收器在被调用时在不同的/新进程上运行。但不同进程之间不支持 SharedPreferences。

所以我所做的是从清单中删除最后一个属性。这意味着代码现在将在主线程上运行 - 但如果您只有几行来显示通知,那么这应该不是问题。另一种方法是调用服务来运行长操作。

于 2012-05-26T22:10:37.027 回答
2

不幸的是,Amir Naor 的解决方案在我的 Android 7 应用程序中不起作用。似乎每个接收者总是在一个新的过程中开始。

API < 23

因此,如果您的应用程序 < API 级别 23,您可以使用以下标志Context.MODE_MULTI_PROCESS

context.getSharedPreferences("mypreferences", Context.MODE_PRIVATE | Context.MODE_MULTI_PROCESS);

API >= 23

真是个惊喜。由于 API 级别 23 已Context.MODE_MULTI_PROCESS 弃用,我们应该使用ContentProvider在进程之间共享属性。


github 上有一个非常不错的库:Tray - Android 的 SharedPreferences 替代品。这个库是一个ContentProvider包含其他一些有用特性的包装器。试试看。

于 2017-01-04T21:09:36.293 回答