Here它表示不推荐使用SimpleCursorAdapter
的API级别1构造函数,建议使用LoaderManager
和CursorLoader
。
但是深入研究了LoaderManager
和CursorLoader
的使用,我发现了this示例,其中一个内部类扩展了ListFragment
(Fragment本身的扩展,我猜想)我们创建一个CursorLoader
。一切似乎都没问题,除了CursorLoader
以Uri
为参数这一事实。所以这意味着我需要创建一个ContentProvider
来访问我的数据库。
我必须承认,为了创建一个包含来自数据库的项目的简单ListView
,我们必须完成所有这些工作。特别是如果我不打算将我的数据库数据提供给其他应用程序,内容提供商的主要目的就是这样做。
那真的值得吗?
特别是在像我这样的情况下,要获取的内容可能会很小。我认真考虑用旧方法做,你说什么?
答案 0 :(得分:8)
我写了一篇不需要内容提供商的simple CursorLoader:
import android.content.Context;
import android.database.Cursor;
import android.support.v4.content.AsyncTaskLoader;
/**
* Used to write apps that run on platforms prior to Android 3.0. When running
* on Android 3.0 or above, this implementation is still used; it does not try
* to switch to the framework's implementation. See the framework SDK
* documentation for a class overview.
*
* This was based on the CursorLoader class
*/
public abstract class SimpleCursorLoader extends AsyncTaskLoader<Cursor> {
private Cursor mCursor;
public SimpleCursorLoader(Context context) {
super(context);
}
/* Runs on a worker thread */
@Override
public abstract Cursor loadInBackground();
/* Runs on the UI thread */
@Override
public void deliverResult(Cursor cursor) {
if (isReset()) {
// An async query came in while the loader is stopped
if (cursor != null) {
cursor.close();
}
return;
}
Cursor oldCursor = mCursor;
mCursor = cursor;
if (isStarted()) {
super.deliverResult(cursor);
}
if (oldCursor != null && oldCursor != cursor && !oldCursor.isClosed()) {
oldCursor.close();
}
}
/**
* Starts an asynchronous load of the contacts list data. When the result is ready the callbacks
* will be called on the UI thread. If a previous load has been completed and is still valid
* the result may be passed to the callbacks immediately.
* <p/>
* Must be called from the UI thread
*/
@Override
protected void onStartLoading() {
if (mCursor != null) {
deliverResult(mCursor);
}
if (takeContentChanged() || mCursor == null) {
forceLoad();
}
}
/**
* Must be called from the UI thread
*/
@Override
protected void onStopLoading() {
// Attempt to cancel the current load task if possible.
cancelLoad();
}
@Override
public void onCanceled(Cursor cursor) {
if (cursor != null && !cursor.isClosed()) {
cursor.close();
}
}
@Override
protected void onReset() {
super.onReset();
// Ensure the loader is stopped
onStopLoading();
if (mCursor != null && !mCursor.isClosed()) {
mCursor.close();
}
mCursor = null;
}
}
它只需要AsyncTaskLoader
类。 Android 3.0或更高版本中的一个,或兼容包附带的那个。
答案 1 :(得分:4)
只需使用它下面的构造函数,即带标记的构造函数。不要使用FLAG_AUTO_REQUERY,只为标志传递0。
除非您真的需要在用户查看ListView时处理对底层数据库的数据更改,否则您无需担心需要重新查询。
另一方面,如果您希望ListView在用户查看列表时显示对数据库的更改,请遵循Google的建议并使用CursorLoader。
修改强>
由于第二个构造函数仅在API 11中可用,您可能只想自己扩展CursorAdapter。你几乎只需要实现bindView和newView,你就完成了。
答案 2 :(得分:1)
仅使用simpleCursorAdapter弃用的构造函数。出现这种错误 我正在开发我的应用程序,但我使用它,它与我的应用程序完美配合。或者尝试在android开发者网站中使用下面不赞成的构造函数,它有一个额外的参数,即带有它的flag参数。
答案 3 :(得分:1)
我认为CursorLoader目前适用于ContentProvider。
如果您希望使用新框架直接从数据库加载;您可以考虑扩展AsyncTaskLoader并从onCreateLoader返回它,而不是使用CursorLoader。
如果您使用现有方法,则必须更加小心查询操作需要多长时间。如果您的查询需要花费大量时间,请考虑使用AsyncTask加载游标(并注意在UI线程中运行的requery)。
答案 4 :(得分:0)
我知道这个线程是旧的,但你可以在SimpleCursorAdapter对象创建中添加最后一个参数。只需添加“,0”。
这是Android喜欢的标志,警告消失了。
示例:
SimpleCursorAdapter dataAdapter = new SimpleCursorAdapter(getApplicationContext(), R.layout.item_list_layout, cursor, fromDB(), toLayout(), 0);