java preference.app_java – 在Android中备份SharedPreferences?

这篇博客探讨了如何在Android中使用SharedPreferencesBackupHelper来实现数据备份和恢复,特别是在应用重新安装后仍能保留数据。作者遇到问题,代码无法在重新安装后读取备份的值。博客内容包括设置BackupAgentHelper,存储和恢复数据的代码示例,以及在AndroidManifest中的配置。然而,备份过程似乎并未按预期工作,作者寻求帮助找出错误并确认备份机制的有效性。

我想在SharedPreferences中备份一个值,以便我可以在重新安装后读出这个值.

我的代码不起作用,我不知道错误是什么.

MyBackupAgent

package com.app.appname;

import android.app.backup.BackupAgentHelper;

import android.app.backup.BackupManager;

import android.app.backup.SharedPreferencesBackupHelper;

import android.content.Context;

public class MyBackupAgent extends BackupAgentHelper{

static final String PREFS_DISPLAY = "AppName";

private Context context;

static final String MY_PREFS_BACKUP_KEY = "keyToStore";

public MyBackupAgent(Context context){

this.context = context;

SharedPreferencesBackupHelper helper =

new SharedPreferencesBackupHelper(context, PREFS_DISPLAY);

addHelper(MY_PREFS_BACKUP_KEY, helper);

}

public void storeData(){

BackupManager backupManager = new BackupManager(context);

backupManager.dataChanged();

}

}

我如何存储数据:

...

SharedPreferences settings = getSharedPreferences("AppName", 0);

SharedPreferences.Editor editor = settings.edit();

editor.putBoolean("keyToStore", true);

editor.commit();

new MyBackupAgent(this).storeData();

...

我如何收到数据:

...

SharedPreferences settings = getSharedPreferences("AppName", 0);

boolean value = settings.getBoolean("keyToStore", false);

...

我还在Android Manifest中添加了API:

你知道我做错了什么以及它是如何工作的吗?它真的有效吗?

这段代码是一个 Android 开发中常见的 `Preference` 点击事件处理方法,通常出现在 `PreferenceActivity` 或 `PreferenceFragment` 的子类中。它的作用是当用户点击某个设置项(`Preference`)时,进行自定义的响应逻辑。 我们来逐行分析并解释其功能: ```java public boolean onPreferenceTreeClick(Preference preference) { if ("ringtone_preference".equals(preference.getKey())) { final Intent intent = new Intent(RingtoneManager.ACTION_RINGTONE_PICKER); intent.putExtra(RingtoneManager.EXTRA_RINGTONE_TYPE, RingtoneManager.TYPE_RINGTONE); intent.putExtra(RingtoneManager.EXTRA_RINGTONE_DEFAULT_URI, RingtoneManager.getDefaultUri(RingtoneManager.TYPE_RINGTONE)); intent.putExtra(RingtoneManager.EXTRA_RINGTONE_SHOW_DEFAULT, false); intent.putExtra(RingtoneManager.EXTRA_RINGTONE_SHOW_SILENT, false); intent.putExtra(RingtoneManager.EXTRA_RINGTONE_TITLE, preference.getTitle()); startActivity(intent); return true; } return super.onPreferenceTreeClick(preference); } ``` --- ### ✅ 代码解释 #### 1. 方法签名 ```java public boolean onPreferenceTreeClick(Preference preference) ``` - 这是 `PreferenceActivity` 或 `PreferenceFragmentCompat` 提供的一个回调方法。 - 当用户点击任意一个 `Preference` 时,系统会调用此方法。 - 返回值为 `true` 表示该点击已被处理,不再传递;返回 `false` 则继续默认处理。 #### 2. 检查点击的是不是特定的 Preference ```java if ("ringtone_preference".equals(preference.getKey())) ``` - 判断当前被点击的 `Preference` 的 `key` 是否是 `"ringtone_preference"`。 - 这个 key 一般在 XML 的 `<Preference>` 标签中定义,例如: ```xml <Preference android:key="ringtone_preference" android:title="选择铃声" /> ``` - 只有点击了这个特定设置项时才执行后续操作。 #### 3. 创建 Intent 启动系统铃声选择器 ```java final Intent intent = new Intent(RingtoneManager.ACTION_RINGTONE_PICKER); ``` - 使用系统内置的 `RingtoneManager.ACTION_RINGTONE_PICKER` 动作启动一个标准的铃声选择界面。 - 这是 Android 提供的标准 API,无需自己实现铃声列表。 #### 4. 设置铃声类型 ```java intent.putExtra(RingtoneManager.EXTRA_RINGTONE_TYPE, RingtoneManager.TYPE_RINGTONE); ``` - 指定要选择的铃声类型:来电铃声(非闹钟、非通知)。 - 其他可选值包括: - `TYPE_NOTIFICATION` - `TYPE_ALARM` - `TYPE_ALL` #### 5. 设置默认铃声 ```java intent.putExtra(RingtoneManager.EXTRA_RINGTONE_DEFAULT_URI, RingtoneManager.getDefaultUri(RingtoneManager.TYPE_RINGTONE)); ``` - 设置当前默认铃声作为“默认”选项的显示值。 - 用户可以选择恢复到默认铃声(虽然下面设置了不显示“默认”选项)。 #### 6. 隐藏“默认”和“静音”选项 ```java intent.putExtra(RingtoneManager.EXTRA_RINGTONE_SHOW_DEFAULT, false); intent.putExtra(RingtoneManager.EXTRA_RINGTONE_SHOW_SILENT, false); ``` - 不显示“使用默认铃声”的选项。 - 不显示“静音”选项。 - 如果你的应用不允许用户设为静音或默认,就可以这样隐藏。 > ⚠️ 注意:即使你传入了这些参数为 `false`,某些设备厂商 ROM 可能仍会显示这些选项。 #### 7. 设置标题 ```java intent.putExtra(RingtoneManager.EXTRA_RINGTONE_TITLE, preference.getTitle()); ``` - 将弹出的铃声选择器窗口的标题设置为当前 `Preference` 的标题(比如“选择铃声”)。 - 提升用户体验一致性。 #### 8. 启动 Activity ```java startActivity(intent); ``` - 跳转到系统的铃声选择界面。 - 但注意:这里直接使用 `startActivity()`,**无法获取用户选择的结果**! > ❗问题所在:如果你想保存用户选择的铃声 URI,必须使用 `startActivityForResult(intent, requestCode)`! #### 9. 返回 true 表示已处理 ```java return true; ``` - 告诉父类不要再处理这个点击事件。 - 若不匹配,则调用 `super.onPreferenceTreeClick(preference)` 让父类处理其他 Preference。 --- ### ❌ 存在的问题(重要改进建议) 当前代码最大的问题是: > **无法获取用户选择的铃声!** 因为使用了 `startActivity(intent)`,而不是 `startActivityForResult(...)`,所以你不知道用户选了哪个铃声。 --- ### ✅ 正确做法(应改为以下方式): 你需要重写 `onActivityResult` 并通过 `startActivityForResult` 获取结果: ```java private static final int REQUEST_RINGTONE = 1; // 在点击时启动 Activity 并请求结果 @Override public boolean onPreferenceTreeClick(Preference preference) { if ("ringtone_preference".equals(preference.getKey())) { Intent intent = new Intent(RingtoneManager.ACTION_RINGTONE_PICKER); intent.putExtra(RingtoneManager.EXTRA_RINGTONE_TYPE, RingtoneManager.TYPE_RINGTONE); intent.putExtra(RingtoneManager.EXTRA_RINGTONE_DEFAULT_URI, RingtoneManager.getDefaultUri(RingtoneManager.TYPE_RINGTONE)); intent.putExtra(RingtoneManager.EXTRA_RINGTONE_SHOW_DEFAULT, false); intent.putExtra(RingtoneManager.EXTRA_RINGTONE_SHOW_SILENT, false); intent.putExtra(RingtoneManager.EXTRA_RINGTONE_TITLE, preference.getTitle()); // 使用 startActivityForResult 替代 startActivity startActivityForResult(intent, REQUEST_RINGTONE); return true; } return super.onPreferenceTreeClick(preference); } // 接收返回结果 @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { if (requestCode == REQUEST_RINGTONE && resultCode == RESULT_OK) { Uri selectedRingtone = data.getParcelableExtra(RingtoneManager.EXTRA_RINGTONE_PICKED_URI); if (selectedRingtone != null) { // 保存选中的铃声,例如存入 SharedPreferences getSharedPreferences("settings", MODE_PRIVATE) .edit() .putString("ringtone_preference", selectedRingtone.toString()) .apply(); // 可更新 Preference 显示文本 } else { // 用户未选择有效铃声 } } } ``` --- ### ✅ 总结 | 功能 | 实现 | |------|------| | 拦截特定 Preference 点击 | ✅ | | 打开系统铃声选择器 | ✅ | | 设置铃声类型与 UI 参数 | ✅ | | **获取用户选择结果** | ❌(原代码缺失)→ 必须用 `startActivityForResult` | --- ###
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符  | 博主筛选后可见
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值