问题 Android:SQLite一对多设计


任何人都对如何实现一对多映射有很好的建议 SQLite 运用 ContentProvider?如果你看看 Uri ContentProvider#insert(Uri, ContentValues) 你可以看到它有 ContentValues 包含要插入的数据的param。问题在于其目前的实施 ContentValues 不支持 put(String, Object) 方法和类是最终的,所以我无法扩展它。为什么这是一个问题?我的设计来了:

我有两个一对多关系的表。为了在代码中表示这些,我有2个模型对象。 1st表示主记录,并且具有一个第二个对象实例列表的字段。现在我在模型对象#1中有一个辅助方法返回 ContentValues 生成当前对象。填充原始字段是微不足道的 ContentValues#put 重载的方法,但我没有运气的清单。所以目前我的第二个表行只是一个字符串值,我生成一个逗号分隔的字符串然后我重新解析为String []里面 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();
    }

}

7870
2018-02-05 06:01


起源

是的,我有同样的问题......如何将多个插入信封包含在一个事务中?我通过添加两个特殊的URI来管理它:一个用于转换的开始,一个用于结束。我知道这不是最好的解决方案。如果你有一个更好的我会欣赏它! - Bhiefer


答案:


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

看看吧 ContactsContract 例如,内容提供商。联系人可以有许多电子邮件地址,许多电话号码等。完成的方式是通过插入/更新/删除 “许多” 侧。要添加新电话号码,请插入新电话号码,提供电话号码所属联系人的ID。

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

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


6
2018-02-05 13:31



好吧,我有一个主要记录,说人参考了很多电话的人。所以我插入Person记录并获取其密钥然后我将bnuch的记录插入到Phone表中,为每个记录提供Person外键。很标准的东西。然后我在Java方面使用我的两个模型对象和Person对象中的Phones集合来表示这种关系没有问题。我确实很难将这些插入Provider API。我打算学习ContactsContract,谢谢你的提示和报告 - Bostone
实际上我的插入不是动态的,基本上我解析XML并插入记录。在那里,我将只阅读这些没有任何进一步的操纵。我想我可以为每个表使用一个ContentResolver但我也希望将此功能公开为ContentProvider,而这又是我在哪里我很难过 - Bostone
实际上我正在通过1.5中的Contacts类 - 再次感谢我的提示,我认为这正是我所需要的 - Bostone
接受这个,因为这是唯一的答案,我得到了启发:) - Bostone
你将如何读取一个游标,该游标包含通过JOIN获得的数据 One-to-Many 关系表? - Muhammad Babar


您可以使用 ContentProviderOperations 为了这。

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

怎么样 ContentProviderOperations 可用于一对多的设计在这个答案中得到了很好的解释: withValueBackReference的语义是什么?


5
2017-12-01 20:04





所以我要回答我自己的问题。我有两个桌子和两个模型对象在正确的轨道上。遗漏了什么,让我感到困惑的是我想直接插入复杂的数据 ContentProvider#insert 在一个电话中。这是错的。 ContentProvider 应该创建和维护这两个表,但是决定使用哪个表应该由Uri参数决定 ContentProvider#insert。使用ContentResolver并向模型对象添加诸如“addFoo”之类的方法非常方便。这样的方法将采用ContentResolver参数,最后这里是插入复杂记录的序列:

  1. 插入父记录 ContentProvider#insert 并获得记录ID
  2. 每个孩子提供父ID(foregn key)并使用 ContentProvider#insert 用不同的Uri插入子记录

所以唯一剩下的问题是如何在交易中包含上述代码?


4
2018-02-05 23:49