6

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

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

  • 该应用程序应在列表中详细显示一些书籍、作者和编辑

  • 该应用程序还应该允许搜索一本书

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

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

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

  • 如果调用已经在执行(比如 toapi/books/1337或 to api/editors),则应该通知调用者它正在加载数据,并且应该给予旧的调用者(如果存在),就好像它是原始调用者一样。

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

问题

在观看并研究了 Virgil Dobjanschi 在 Google I/O 2010 上提出的所有组件之后,我的疑惑是:

  1. 如何透明地处理任何调用者的“实体正在更新”概念?我应该ContentObserverContentProvider我必须实施的情况下使用吗?

  2. 如果我使用 aContentObserver我可以轻松地为单个实体设置状态标志(如 Dobjanschi 所建议的那样),例如UPDATINGINSERTING等。但是我应该如何处理列表?假设我想要一个书籍清单,我应该把状态标志放在哪里?我应该将它放在仅用于列表的状态表中吗?如果是这样,我应该观察两个Cursors,一个用于状态,一个用于实际列表(即 table/Content URI)。如果我要求的实体不存在(还)或者 REST 调用返回一个404? 如何处理回调?

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

  4. 可以SyncAdapter多个实体上工作(实际上是实体列表,因为我想不时更新书籍和编辑器),因为它只有一个performSync方法?

  5. 如果SyncAdapter用户在设备设置中禁用了我的实现,它将不会更新任何内容(这很好)。但是,如果用户在 Activity 中单击“更新书籍”按钮,我仍然可以调用该performSync方法,还是它也会被禁用?

4

2 回答 2

10

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

  1. 一个应用程序。这使用了一组Activitywith Cursorand ContentObserverand might CursorAdapterand some 来为本地存储的数据提供一个 UI 来自ContentProvider.
  2. ContentProvider 本地设备的数据存储。处理 CRUD 调用,处理SyncAdapter需要将更新推送到服务器的通知。
  3. Account远程服务器上的用户身份。
  4. SyncAdapter一个后台进程,运行并保持本地数据存储与服务器同步。
  5. 服务器本身。

所以。对于问题:

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

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

  3. 一个。为了确保将本地更新发送到服务器,您编写 ContentProvider 以在 ContentProvider 的 Create/Update/Delete 调用期间通知 SyncAdapter。(在这里阅读...)
    b。为确保您定期从服务器获取更新,请将帐户配置为自动同步。 (在这里阅读...)

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

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

于 2012-09-18T14:58:52.393 回答