如何处理REST调用,数据持久性,同步和观察ContentProvider

时间:2012-06-22 16:22:25

标签: android rest android-contentprovider android-syncadapter contentobserver

我知道这个问题被问了太多次,但我认为我试图解决的问题有点不同,可能更复杂。

我将开发一个使用 RESTful Web服务的应用程序,并且需要满足以下要求:

  • 该应用应在列表和详细信息中显示一些图书,作者及其编辑

  • 该应用还应该允许搜索图书

  • 书籍,作者和编辑者从RESTful Web服务中获取

  • 每个实体都必须进行缓存,这样当我打开一个Activity时,我会首先看到旧数据(如果有的话),而新的数据会从网络更新。

  • 每次实体更新时,都应通知相关方(ContentObserver?定期Listener实施?)

  • 如果呼叫已在执行(例如对api/books/1337api/editors),则应通知呼叫者正在加载数据,并且应该给出旧呼叫(如果存在) ,好像是原来的来电者。

  • 某些数据(只有书籍和作者)应每N分钟更新一次(由用户决定),并应通知观察员(SyncAdapter?)

问题

在观察和研究所有组件proposed by Virgil Dobjanschi at Google I/O 2010后,我怀疑:

  1. 如何透明地处理任何来电者的“实体更新”概念?我应该ContentObserver使用ContentProvider我必须实施吗?

  2. 如果我使用 ContentObserver ,我可以轻松地为单个实体设置状态标记(如Dobjanschi所建议的),例如UPDATING,{ {1}},等等。但是我该如何处理清单呢?说我想要一本书清单,我应该把状态标志放在哪里?我应该将它放在状态表中仅列表吗?如果是这样,我应该观察两个INSERTING,一个用于状态,一个用于实际列表(即表/内容URI)。如果我要求的实体(尚未存在)或REST调用返回Cursor该怎么办?我该如何处理回调?

  3. 如果我将所有REST方法放在404中,我可以“强制”**SyncAdapter**从网络更新实体/实体列表(并因此将其放入适当的表)?这样,状态标志就会很有用。

  4. SyncAdapter可以使用多个实体(实际上,实体列表,因为我想不时更新图书和编辑器),因为它只有{ {1}}方法?

  5. 如果用户在设备设置中禁用了我的SyncAdapter实现,则不会更新任何内容(这没关系)。但是,如果用户点击了活动中的“更新图书”按钮,我仍然可以调用performSync方法,还是也会被禁用?

2 个答案:

答案 0 :(得分:10)

SyncAdapter是一个涉及五个组件的设计模式:

  1. 一个应用程序。这会使用一组Activity以及CursorContentObserver以及CursorAdapter和一些来为ContentProvider中的本地存储数据提供UI。 / LI>
  2. ContentProvider本地设备的数据存储。处理CRUD调用,处理通知SyncAdapter需要将更新推送到服务器。
  3. Account远程服务器上的用户身份。
  4. SyncAdapter后台进程,它运行并使本地数据存储与服务器保持同步。
  5. 服务器本身。
  6. 因此。问题:

    1. “正在更新”意味着“具有尚未推送到服务器的本地更改。它是您在数据库中的行上设置的标志。当您创建/更新/删除时,它在ContentProvider中设置当SyncAdapter运行时,它会看到标志,将更新推送到服务器,清除标志。标志本身有两件事:
      a。告诉用户应用正忙于保存更改,以及何时完成 b。将行标记为已更改,以便SyncAdapter知道将其推送到服务器 Read here了解更多详情。

    2. 如果您没有同步整个目录,那么您的客户端将直接查询服务器并通过将结果放入ContentProvider来缓存结果。那里没有状态标志,因为它们来自服务器,因此与服务器状态匹配。编写你的SyncAdapter以忽略它们,或者在它们被缓存几天后丢弃它们。

    3. a。要确保将本地更新发送到服务器,请编写ContentProvider以在ContentProvider的创建/更新/删除调用期间通知SyncAdapter。 (Read here...)
      b。为确保您定期从服务器获取更新,请将帐户配置为自动同步。 (Read Here...)

    4. 是。 performSync只是一个函数调用。写它来做你想要的。让它从服务器获取表1并将其放入ContentProvider中的一个表中。然后让它获取表2,并将其放入另一个表中。等

    5. a。您可以通过在附加包中使用ContentResolver.RequestSync()调用ContentResolver.SYNC_EXTRAS_MANUAL来强制同步。
      b。您可以使用客户端代码手动获取内容并直接将其推送到ContentProvider。

答案 1 :(得分:0)