11

有人对如何实现一对多映射有很好的建议SQLiteContentProvider?如果您查看,Uri ContentProvider#insert(Uri, ContentValues)您会发现它具有ContentValues包含要插入的数据的参数。问题是它当前的实现ContentValues不支持put(String, Object)方法并且类是最终的,所以我不能扩展它。为什么这是一个问题?我的设计来了:

我有 2 个表,它们是一对多的关系。为了在代码中表示这些,我有 2 个模型对象。1st 代表主记录,并有一个字段是第二个对象实例的列表。现在我在模型对象#1 中有一个辅助方法,它返回ContentValues从当前对象生成的。用重载方法填充原始字段很简单,ContentValues#put但我对列表不走运。所以目前由于我的第二个表行只是一个字符串值,我生成了一个逗号分隔的字符串,然后我将其重新解析为 String[] inside ContentProvider#insert。这感觉很糟糕,所以也许有人可以暗示如何以更清洁的方式完成它。

这是一些代码。首先来自模型类:

public ContentValues toContentValues() {
    ContentValues values = new ContentValues();
    values.put(ITEM_ID, itemId);
    values.put(NAME, name);
    values.put(TYPES, concat(types));
    return values;
}

private String concat(String[] values) { /* trivial */}

ContentProvider#insert这是方法的精简版

public Uri insert(Uri uri, ContentValues values) {
    SQLiteDatabase db = dbHelper.getWritableDatabase();
    db.beginTransaction();
    try {
        // populate types
        String[] types = ((String)values.get(Offer.TYPES)).split("|");
        // we no longer need it
        values.remove(Offer.TYPES);
        // first insert row into OFFERS
        final long rowId = db.insert("offers", Offer.NAME, values);
        if (rowId > 0 && types != null) {
            // now insert all types for the row
            for (String t : types) {
                ContentValues type = new ContentValues(8);
                type.put(Offer.OFFER_ID, rowId);
                type.put(Offer.TYPE, t);
                // insert values into second table
                db.insert("types", Offer.TYPE, type);
            }
        }
        db.setTransactionSuccessful();
        return ContentUris.withAppendedId(Offer.CONTENT_URI, rowId);
    } catch (Exception e) {
        Log.e(TAG, "Failed to insert record", e);
    } finally {
        db.endTransaction();
    }

}
4

3 回答 3

6

我认为您正在寻找一对多关系的错误结尾。

例如,看一下ContactsContract内容提供者。联系人可以有许多电子邮件地址、许多电话号码等。完成的方法是在“多”端进行插入/更新/删除。要添加新电话号码,请插入一个新电话号码,并提供与该电话号码相关的联系人的 ID。

如果你有一个没有内容提供者的普通 SQLite 数据库,你也会这样做。关系数据库中的一对多关系是通过在“多”端的表上插入/更新/删除来实现的,每个表都有一个返回“一”端的外键。

现在,从面向对象的角度来看,这并不理想。欢迎您创建 ORM 样式的包装对象(想想 Hibernate),它允许您从“一”端操作一组子对象。然后,一个足够智能的集合类可以转身并同步“many”表以进行匹配。但是,正确实施这些并不一定是微不足道的。

于 2010-02-05T13:31:09.783 回答
5

你可以用ContentProviderOperations这个。

它们基本上是批量操作,能够反向引用为父行生成的标识符。

这个答案很好地解释了如何ContentProviderOperations用于一对多设计:withValueBackReference 的语义是什么?

于 2011-12-01T20:04:36.660 回答
4

所以我要回答我自己的问题。我在正确的轨道上拥有两个表和两个模型对象。缺少的和让我困惑的是,我想ContentProvider#insert在一次调用中直接插入复杂的数据。这是错误的。ContentProvider应该创建和维护这两个表,但决定使用哪个表应该由 的 Uri 参数决定ContentProvider#insert。使用 ContentResolver 并在模型对象中添加“addFoo”等方法非常方便。这种方法将采用 ContentResolver 参数,最后是插入复杂记录的序列:

  1. 通过插入父记录ContentProvider#insert并获取记录id
  2. 每个孩子提供父 ID(外键)并使用ContentProvider#insert不同的 Uri 插入子记录

那么剩下的唯一问题是如何将上述代码封装在事务中?

于 2010-02-05T23:49:42.387 回答