- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我一直在组装一个简单的 Android ListView 应用程序来自学 LoaderManagers 和 Loaders。该应用仅提供一个加载器,当共享首选项发生变化时,该加载器会通过 OnSharedPreferenceChangeListener 收到通知。
它有一个带有 ListView 和两个按钮的简单 UI,一个用于添加首选项,一个用于删除所有共享首选项。
我的 Activity 是 ListActivity 的子类并实现了 LoaderManager.LoaderCallbacks>。
My Loader 仅在共享首选项发生更改时通知 Activity。
我的问题是,当单击删除所有首选项按钮时,不会通知 OnSharedPreferenceChangeListener,而添加首选项会通知它。即使没有通知收听者,偏好也会被删除,因为随后单击添加偏好按钮会更新列表以显示单个列表条目。
这是代码(为简洁起见省略了一些位):
主 Activity .java
public class MainActivity extends ListActivity implements LoaderManager.LoaderCallbacks<List<String>>
{
private ListView listView;
private static final int LOADER_ID = 1;
private LoaderManager.LoaderCallbacks<List<String>> mCallbacks;
private ArrayAdapter<String> adapter;
@Override
protected void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
setUpLoaderManager();
ArrayList<String> values = new ArrayList<String>();
adapter = new ArrayAdapter<String>(this, android.R.layout.simple_list_item_1, android.R.id.text1, values);
listView = (ListView) findViewById(android.R.id.list);
listView.setAdapter(adapter);
}
private void setUpLoaderManager()
{
mCallbacks = this;
LoaderManager lm = getLoaderManager();
lm.initLoader(LOADER_ID, null, mCallbacks);
}
@Override
public void onLoadFinished(Loader<List<String>> loader, List<String> data)
{
switch (loader.getId())
{
case LOADER_ID:
adapter.clear();
adapter.addAll(data);
break;
}
}
@Override
public Loader<List<String>> onCreateLoader(int id, Bundle args)
{
return new SampleLoader(this);
}
public void addSharedPreference(View v)
{
getSharedPreferences("someData", Context.MODE_PRIVATE).edit().putString(new Date().toString(), "hi").commit();
}
public void deleteSharedPreference(View v)
{
SharedPreferences sharedPreferences = getSharedPreferences("someData", Context.MODE_PRIVATE);
sharedPreferences.edit().clear().commit();
}
示例加载器.java
public class SampleLoader extends AsyncTaskLoader<List<String>>
{
private List<String> mData;
public SampleLoader(Context ctx)
{
super(ctx);
}
@Override
public List<String> loadInBackground()
{
List<String> data = new ArrayList<String>();
SharedPreferences prefs = getContext().getSharedPreferences("someData", Context.MODE_PRIVATE);
Map<String, String> currentPref = (Map<String, String>) prefs.getAll();
if (currentPref.size() > 0)
{
for (Map.Entry<String, String> entry : currentPref.entrySet())
{
data.add(entry.getKey());
}
}
return data;
}
@Override
public void deliverResult(List<String> data)
{
if (isReset())
{
releaseResources(data);
return;
}
List<String> oldData = mData;
mData = data;
if (isStarted())
{
super.deliverResult(data);
}
if (oldData != null && oldData != data)
{
releaseResources(oldData);
}
}
@Override
protected void onStartLoading()
{
if (mData != null)
{
deliverResult(mData);
}
if (mObserver == null)
{
mObserver = new SharedPrefsObserver(this, getContext());
}
if (takeContentChanged() || mData == null)
{
forceLoad();
}
}
@Override
protected void onStopLoading()
{
cancelLoad();
}
@Override
protected void onReset()
{
onStopLoading();
if (mData != null)
{
releaseResources(mData);
mData = null;
}
if (mObserver != null)
{
mObserver = null;
}
}
@Override
public void onCanceled(List<String> data)
{
super.onCanceled(data);
releaseResources(data);
}
private void releaseResources(List<String> data)
{
}
private SharedPrefsObserver mObserver;
}
SharedPrefsObserver.java
public class SharedPrefsObserver
{
public static OnSharedPreferenceChangeListener listener;
public SharedPrefsObserver(final SampleLoader sampleLoader, Context context)
{
final SharedPreferences prefs = context.getSharedPreferences("someData", Context.MODE_PRIVATE);
listener = new OnSharedPreferenceChangeListener()
{
@Override
public void onSharedPreferenceChanged(SharedPreferences sharedPreferences, String key)
{
sampleLoader.onContentChanged();
}
};
prefs.registerOnSharedPreferenceChangeListener(listener);
}
}
activity_main.xml
<LinearLayout
android:id="@+id/button_layout"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:layout_alignParentBottom="true"
android:weightSum="2">
<Button
android:id="@+id/button_delete"
android:layout_width="0dp"
android:layout_weight="1"
android:layout_height="wrap_content"
android:onClick="deleteSharedPreference"
android:text="Delete all preferences" >
</Button>
<Button
android:id="@+id/button_add"
android:layout_width="0dp"
android:layout_weight="1"
android:layout_height="wrap_content"
android:onClick="addSharedPreference"
android:text="Add new preference" >
</Button>
</LinearLayout>
<ListView
android:id="@android:id/list"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_alignParentTop="true"
android:layout_above="@id/button_layout"
android:layout_alignParentLeft="true">
</ListView>
这是我不得不用来强制加载器在删除首选项后刷新数据的解决方法:
public void deleteSharedPreference(View v)
{
SharedPreferences sharedPreferences = getSharedPreferences("someData", Context.MODE_PRIVATE);
sharedPreferences.edit().clear().commit();
//dirty hack - sharedPreferences observer is not notified when preferences are removed so had to bypass it and call the loader manually
getLoaderManager().getLoader(LOADER_ID).forceLoad();
}
那么,关于为什么删除首选项没有通知我的听众有什么想法吗?
最佳答案
sharedPreferences.edit().clear().commit()
按照设计不会通知监听器。如果您希望收到删除共享首选项的通知,则需要使用 remove(key)
而不是 clear()
。
查看 Android source ,你可以看到 clear()
和 remove(key)
的实现略有不同。当 remove(key)
将键添加到由编辑器创建的私有(private) HashMap 对象以跟踪更改时,clear()
只是设置一个标志并且不进行任何更改到 HashMap 对象。 SharedPreferences 私有(private)方法 NotifyListeners()不尊重这个标志,它只看 HashMap 对象。所以调用clear()
不会被监听器报告。
[编辑]:删除 key 时,您也必须使用相同的 editor
实例来提交它:
SharedPreferences settings = getSharedPreferences("someData", MODE_PRIVATE);
SharedPreferences.Editor editor = settings.edit();
editor.remove(myKey1);
editor.remove(myKey2);
...
editor.commit();
关于android - 从 SharedPreferences 中删除条目后未通知 OnSharedPreferenceChangeListener,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24852293/
我想知道以跨平台方式操作应用程序设置的最佳解决方案是什么。 在 iOS 中,我们可以在设置屏幕中更改应用程序外部的设置,但在 windows phone 和 android 中我们没有。 所以,我的想
我想清除除一个 SharedPreference 之外的所有内容。如果我已经保存了 10 个以上,有没有比单独删除每个更好的方法?这有点多余: preferences.edit().remove("1
我在名为 page1.dart 的页面中存储了一个值。我想从 page2.dart 或 page3.dart 访问存储的值。我怎样才能做到这一点? 最佳答案 Flutter 共享首选项实际上是作为内存
是否可以将 SharedPreferences 文件保存到自定义目录中?让我们说到 /data/data/package.name/my_prefs/。 或者是否可以检索默认保存 SharedPref
我很好奇存储库在 MVVM 架构中的作用。如果您决定将存储库添加到您的项目中,这个存储库是否只负责来自数据库或网络的数据?问题是关于 SharedPreferences 或 Files,我应该让存储库
尝试从服务创建共享首选项文件时出现以下错误: “无法为 SharedPreferences 文件/dbdata/databases/dimappers.android.pub/shared_prefs
所以我有这行代码 SharedPreferences sharedPref = getApplicationContext().getSharedPreferences(PREFS_NAME, 0);
This question already has answers here: What is a NullPointerException, and how do I fix it? (12个答案)
有谁知道 android 是否支持在一个项目中编译的多个 Android 模块之间共享相同的 sharedpreference? 我有两个共享首选项,目前当我尝试从当前模块外部的共享首选项访问一些数据
我正在搜索最大的 Android SharedPreferences 键值对,但找不到任何好的答案。其次,我想问一下,如果我有一个键,它的字符串值限制是多少。多少字符可以放入其中。如果我需要频繁更改值
我正在制作一个简单的预算跟踪应用程序。我有一个预算类,其中存储了一些变量:预算的金额、持续时间和初始值。我在 fragment 中只有一个全局预算对象,称为“预算”,我正在尝试保存它。它似乎保存得很好
我在 Flutter 应用程序中使用了 shared_preferences 插件,我想在用户选择城市时保存数据。 但是当我尝试打印它时,它只是说; Instance of 'SharedPrefer
我阅读了有关在 Flutter 应用程序中初始化 SharedPreferences 的其他答案(例如 this one ),但我想采取另一条路线:一开始就初始化 SharedPreferences
在我的 flutter 应用程序中,我实现了一个入职 View 。因为它应该只加载一次,所以我使用共享首选项来存储一个整数来表示已经显示了入职。当我在 Debug模式下运行应用程序时,一切正常。但是当
如果您使用 apply,在单独的线程中编辑共享首选项是否多余? 我的 MainActivity 的 onCreate 方法中有以下代码块: final MainActivity activit
我有 2 个 Activity ,身份验证和主页。 在身份验证中,它检查用户是否已登录,如果用户已登录,则将其重定向到Mainpage.class。这就是身份验证检查用户是否登录并重定向的方式。 Sh
为什么 toast 只显示空字符串或我在 getString 行上输入的“默认”值 @Override protected void onCreate(Bundle savedInstanceStat
我在我的项目中创建了一个模块。该模块的名称为“连接器”。现在我在 MainActivity 中创建 private SharedPreferences sPref; private S
刚接触 Java,很抱歉我的理解不够。我遇到了一个小障碍,我有一个用于连接到服务器的静态类。我正在为 SharedPreferences 使用另一个类,SharedPreferences 的一些详细信
这是一件好事。我将 int 值存储在我的共享首选项中,本质上是缓存它们,以便我可以在下次启动时将它们放入数据库中。我有一种方法可以在启动时将它们保存到数据库中。 接下来是从共享首选项中删除值以重置它们
我是一名优秀的程序员,十分优秀!