Android CursorLoader> ContentProvider> SqliteDatabase无法正常工作

时间:2013-06-26 15:14:00

标签: android sqlite android-contentprovider android-cursorloader

我已经学了几个教程,解释了如何将CursorLoader与SQLiteDatabase一起使用。当我运行ListView时,没有返回任何结果。我已经确认使用Eclipse插件在数据库中有一个条目。

以下是我使用sqliteopenhelper实现的contentprovider

public class DiaryDataProvider extends ContentProvider {

public static final String DATABASE_NAME = "data.db";
public static final int DATABASE_VERSION = 1;

// implementation of SQLiteOpenHelper
private static class DiaryDbHelper extends SQLiteOpenHelper {
    // creating and deleting the database and table
    private static final String TEXT_TYPE = " TEXT";
    private static final String REAL_TYPE = " REAL";
    private static final String INTEGER_TYPE = " INTEGER";
    private static final String COMMA_SEP = ",";

    private static final String SQL_CREATE_ENTRIES = "CREATE TABLE "
            + DiarySchema.TABLE_NAME + " (" + DiarySchema._ID
            + " INTEGER PRIMARY KEY," + DiarySchema.COLUMN_NAME_DIARY_ID
            + TEXT_TYPE + COMMA_SEP + DiarySchema.COLUMN_NAME_DATE
            + TEXT_TYPE + COMMA_SEP + DiarySchema.COLUMN_NAME_TIME
            + TEXT_TYPE + COMMA_SEP + DiarySchema.COLUMN_NAME_SEVERITY
            + TEXT_TYPE + COMMA_SEP + DiarySchema.COLUMN_NAME_LOCATION_LAT
            + REAL_TYPE + COMMA_SEP + DiarySchema.COLUMN_NAME_LOCATION_LONG
            + REAL_TYPE + COMMA_SEP
            + DiarySchema.COLUMN_NAME_LOCATION_DETAIL + TEXT_TYPE
            + COMMA_SEP + DiarySchema.COLUMN_NAME_HAS_FOOD + INTEGER_TYPE
            + COMMA_SEP + DiarySchema.COLUMN_NAME_DETAILS + TEXT_TYPE
            + " )";

    public DiaryDbHelper(Context context) {
        super(context, DATABASE_NAME, null, DATABASE_VERSION);
    }

    @Override
    public void onCreate(SQLiteDatabase db) {
        db.execSQL(SQL_CREATE_ENTRIES);
    }

    @Override
    public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
        db.execSQL("DROP TABLE IF EXISTS " + DiarySchema.TABLE_NAME);
        onCreate(db);
    }
}

private DiaryDbHelper dbHelper;

@Override
public int delete(Uri uri, String selection, String[] selectionArgs) {
    SQLiteDatabase db = dbHelper.getReadableDatabase();
    final int match = DiarySchema.URI_MATCHER.match(uri);
    switch (match) {
    // retrieve diary list
    case DiarySchema.PATH_TOKEN: {
        SQLiteQueryBuilder builder = new SQLiteQueryBuilder();
        builder.setTables(DiarySchema.TABLE_NAME);
        return db.delete(DiarySchema.TABLE_NAME, selection, selectionArgs);
    }
    default:
        return 0;
    }

}

@Override
public String getType(Uri uri) {
    final int match = DiarySchema.URI_MATCHER.match(uri);
    switch (match) {
    case DiarySchema.PATH_TOKEN:
        return DiarySchema.CONTENT_TYPE_DIR;
    case DiarySchema.PATH_FOR_ID_TOKEN:
        return DiarySchema.CONTENT_ITEM_TYPE;
    default:
        throw new UnsupportedOperationException("URI " + uri
                + " is not supported.");
    }
}

@Override
public Uri insert(Uri uri, ContentValues values) {
    SQLiteDatabase db = dbHelper.getWritableDatabase();
    int token = DiarySchema.URI_MATCHER.match(uri);
    switch (token) {
    case DiarySchema.PATH_TOKEN: {
        long id = db.insert(DiarySchema.TABLE_NAME, null, values);
        getContext().getContentResolver().notifyChange(uri, null);
        return DiarySchema.CONTENT_URI.buildUpon()
                .appendPath(String.valueOf(id)).build();
    }
    default: {
        throw new UnsupportedOperationException("URI: " + uri
                + " not supported.");
    }
    }
}

@Override
public boolean onCreate() {
    dbHelper = new DiaryDbHelper(getContext());
    return true;
}

@Override
public Cursor query(Uri uri, String[] projection, String selection,
        String[] selectionArgs, String sortOrder) {
    SQLiteDatabase db = dbHelper.getReadableDatabase();
    final int match = DiarySchema.URI_MATCHER.match(uri);
    switch (match) {
    // retrieve diary list
    case DiarySchema.PATH_TOKEN: {
        SQLiteQueryBuilder builder = new SQLiteQueryBuilder();
        builder.setTables(DiarySchema.TABLE_NAME);
        return builder.query(db, projection, selection, null, null, sortOrder, null);
    }
    default:
        return null;
    }
}

@Override
public int update(Uri uri, ContentValues values, String selection,
        String[] selectionArgs) {
    SQLiteDatabase db = dbHelper.getReadableDatabase();
    final int match = DiarySchema.URI_MATCHER.match(uri);
    switch (match) {
    // retrieve diary list
    case DiarySchema.PATH_TOKEN: {
        SQLiteQueryBuilder builder = new SQLiteQueryBuilder();
        builder.setTables(DiarySchema.TABLE_NAME);
        return db.update(DiarySchema.TABLE_NAME, values, selection,
                selectionArgs);
    }
    default:
        return 0;
    }

}
}

这是随之而来的架构

public abstract class DiarySchema implements BaseColumns {
public static final String TABLE_NAME = "diary";
public static final String COLUMN_NAME_DIARY_ID = "diaryid";
public static final String COLUMN_NAME_DATE = "date";
public static final String COLUMN_NAME_TIME = "time";
public static final String COLUMN_NAME_SEVERITY = "severity";
public static final String COLUMN_NAME_LOCATION_LAT = "locationlat";
public static final String COLUMN_NAME_LOCATION_LONG = "locationlong";
public static final String COLUMN_NAME_LOCATION_DETAIL = "locationdetail";
public static final String COLUMN_NAME_DETAILS = "details";
public static final String COLUMN_NAME_HAS_FOOD = "hasfood";

public static final String PATH = "diaries";
public static final int PATH_TOKEN = 100;
public static final String PATH_FOR_ID = "diaries/#";
public static final int PATH_FOR_ID_TOKEN = 200;

public static final String CONTENT_TYPE_DIR = ContentResolver.CURSOR_DIR_BASE_TYPE + "/diaries";
public static final String CONTENT_ITEM_TYPE = ContentResolver.CURSOR_ITEM_BASE_TYPE + "/diaries";

public static final String AUTHORITY = "com.foo.bar.contentproviders";
private static final Uri BASE_URI = Uri.parse("content://" + AUTHORITY);
public static final UriMatcher URI_MATCHER = buildUriMatcher();
public static final Uri CONTENT_URI = BASE_URI.buildUpon().appendPath(PATH)
        .build();

private static UriMatcher buildUriMatcher() {
    final UriMatcher matcher = new UriMatcher(UriMatcher.NO_MATCH);
    final String authority = AUTHORITY;

    matcher.addURI(authority, PATH, PATH_TOKEN);
    matcher.addURI(authority, PATH_FOR_ID, PATH_FOR_ID_TOKEN);

    return matcher;
}

// Prevent anyone from instantiating this class
private DiarySchema() {
};
}

我的ContentProvider在androidmanifest.xml文件中声明为...

<provider
        android:name="com.foo.bar.contentproviders.DiaryDataProvider"
        android:authorities="com.foo.bar.contentproviders"
        android:exported="false" />

我的ListFragment看起来像......

public class FragDiaryEntries extends SherlockListFragment implements
    LoaderManager.LoaderCallbacks<Cursor> {
private static final int URL_LOADER = 0;
private View view;
private Map<Integer, DiaryHolder> diaryHolders = new HashMap<Integer, DiaryHolder>();
CustomAdapter adapter;

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
        Bundle savedInstanceState) {

    view = inflater.inflate(R.layout.frag_diary_entries, container);

    // start the cursor loader getting data to populate diary view
    getLoaderManager().initLoader(URL_LOADER, null, this);

    // Set up adapter
    adapter = new CustomAdapter(getActivity().getApplicationContext(), null);
    setListAdapter(adapter);

    return view;
}

@Override
public void onActivityCreated(Bundle savedInstanceState) {
    // TODO different indicator of no data yet
    ((TextView) getListView().getEmptyView()).setText("It's Empty");

    super.onActivityCreated(savedInstanceState);
}



/* Loader callback implemented methods */

@Override
public Loader<Cursor> onCreateLoader(int loaderId, Bundle bundle) {
    Loader<Cursor> cursorLoader = null;

    // Get all of the diary entries
    switch (loaderId) {
    case URL_LOADER:
        String[] projection = { DiarySchema._ID,
                DiarySchema.COLUMN_NAME_DATE, DiarySchema.COLUMN_NAME_TIME,
                DiarySchema.COLUMN_NAME_SEVERITY,
                DiarySchema.COLUMN_NAME_LOCATION_LAT,
                DiarySchema.COLUMN_NAME_LOCATION_LONG,
                DiarySchema.COLUMN_NAME_LOCATION_DETAIL,
                DiarySchema.COLUMN_NAME_DETAILS,
                DiarySchema.COLUMN_NAME_HAS_FOOD };

        cursorLoader = new CursorLoader(getActivity(),
                DiarySchema.CONTENT_URI, projection, null, null, null);

    }
    return cursorLoader;
}

@Override
public void onLoadFinished(Loader<Cursor> loader, Cursor cursor) {
    adapter.swapCursor(cursor);
}

@Override
public void onLoaderReset(Loader<Cursor> loader) {
    adapter.swapCursor(null);
}

// Custom CursorAdapter
private class CustomAdapter extends CursorAdapter {

    private Cursor cursor;

    public CustomAdapter(Context context, Cursor c) {
        super(context, c, 0);
        this.cursor = c;
    }

    @Override
    public int getCount() {
        return cursor == null ? 0 : cursor.getCount();
    }

    @Override
    public Cursor getCursor() {
        return super.getCursor();
    }

    @Override
    public void bindView(View view, Context context, Cursor cursor) {

        int position = cursor.getPosition();

        // Map values from cursor to fields
        TextView created = (TextView) view.findViewById(R.id.diary_created);
        TextView severity = (TextView) view
                .findViewById(R.id.diary_severity);
        TextView location = (TextView) view
                .findViewById(R.id.diary_location);
        CheckBox checkbox = (CheckBox) view
                .findViewById(R.id.diary_item_checked);

        // Create a holder object to store some row details
        if (!diaryHolders.containsKey(position)) {
            // Put the cursor in the correct position
            cursor.moveToPosition(position);

            String date = cursor.getString(cursor
                    .getColumnIndexOrThrow(DiarySchema.COLUMN_NAME_DATE));
            String time = cursor.getString(cursor
                    .getColumnIndexOrThrow(DiarySchema.COLUMN_NAME_TIME));
            created.setText(date + " at " + time);

            String sev = cursor
                    .getString(cursor
                            .getColumnIndexOrThrow(DiarySchema.COLUMN_NAME_SEVERITY));
            severity.setText(sev);

            String loc = cursor
                    .getString(cursor
                            .getColumnIndexOrThrow(DiarySchema.COLUMN_NAME_LOCATION_DETAIL));
            location.setText(loc);

            // Other values to put into holder
            long latitude = cursor
                    .getLong(cursor
                            .getColumnIndexOrThrow(DiarySchema.COLUMN_NAME_LOCATION_LAT));
            long longitude = cursor
                    .getLong(cursor
                            .getColumnIndexOrThrow(DiarySchema.COLUMN_NAME_LOCATION_LONG));
            int hasFood = cursor
                    .getInt(cursor
                            .getColumnIndexOrThrow(DiarySchema.COLUMN_NAME_HAS_FOOD));

            diaryHolders.put(position,
                    new DiaryHolder(position, checkbox.isChecked(),
                            new Diary(date, time, sev, loc, latitude,
                                    longitude, hasFood)));
        } else {

            // Populate from map
            DiaryHolder holder = diaryHolders.get(position);
            Diary diary = holder.getDiaryData();

            created.setText(diary.getDate().concat(diary.getTime()));
            severity.setText(diary.getSeverity());
            location.setText(diary.getLocationDetail());
            checkbox.setChecked(holder.isSelected());
        }

    }

    @Override
    public View newView(Context context, Cursor cursor, ViewGroup container) {

        // Inflate a new view
        LayoutInflater inflater = LayoutInflater.from(context);
        View view = inflater.inflate(R.layout.diary_list_item, container);
        bindView(view, context, cursor);
        return view;
    }

}

// Holder class for row values
private class DiaryHolder {
    private long rowId;
    private boolean selected;
    private Diary diaryData;

    public DiaryHolder(long rowId, boolean selected, Diary diaryData) {
        this.rowId = rowId;
        this.selected = selected;
        this.diaryData = diaryData;
    }

    public Diary getDiaryData() {
        return diaryData;
    }

    public void setDiaryData(Diary diaryData) {
        this.diaryData = diaryData;
    }

    public long getRowId() {
        return rowId;
    }

    public void setRowId(long rowId) {
        this.rowId = rowId;
    }

    public boolean isSelected() {
        return selected;
    }

    public void setSelected(boolean selected) {
        this.selected = selected;
    }

}

}

我知道我的ContentProvider的插入部分有效,因为我能够插入记录。我想要工作的是我将所有数据拉回并在listview中显示。任何帮助将不胜感激。

2 个答案:

答案 0 :(得分:0)

我没有看到你在返回之前在光标上调用setNotificatioUri

查看我认识的一种查询方法

@Override
public Cursor query(Uri uri, String[] projection, String selection,
        String[] selectionArgs, String sortOrder) {
    SQLiteQueryBuilder sqlBuilder = new SQLiteQueryBuilder();
    sqlBuilder.setTables(CAL_EVENTS_TABLE);

    if (uriMatcher.match(uri) == 1) {
        sqlBuilder.setProjectionMap(mMap);
    } else if (uriMatcher.match(uri) == 2) {
        sqlBuilder.setProjectionMap(mMap);
        sqlBuilder.appendWhere(ID + "=?");
        selectionArgs = DatabaseUtils.appendSelectionArgs(selectionArgs,
                new String[] { uri.getLastPathSegment() });
    }
    if (sortOrder == null || sortOrder == "")
        sortOrder = START_DATE + " COLLATE LOCALIZED ASC";
    Cursor c = sqlBuilder.query(db, projection, selection, selectionArgs,
            null, null, sortOrder);
    c.setNotificationUri(getContext().getContentResolver(), uri);
    return c;
}

这里的代码特别是

if (sortOrder == null || sortOrder == "")
    sortOrder = START_DATE + " COLLATE LOCALIZED ASC";
Cursor c = sqlBuilder.query(db, projection, selection, selectionArgs,
        null, null, sortOrder);
c.setNotificationUri(getContext().getContentResolver(), uri);

正如您所看到的,我打电话给setNotificationUri,文档说的是

Register to watch a content URI for changes. This can be the URI of a specific data row (for example, "content://my_provider_type/23"), or a a generic URI for a content type.

并且ContentResolver参数显示

The content resolver from the caller's context. The listener attached to this resolver will be notified.

答案 1 :(得分:0)

好吧,伙计们我找到了答案。我已经覆盖了CustomAdapter的getCursor()方法,它返回了super.getCursor();显然(duh)当我实例化我的适配器对象时,super的光标被设置为null,所以它总是返回null:)

感谢你的所有建议。