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

fra*_*llo 6 rest android android-contentprovider android-syncadapter contentobserver

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

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

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

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

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

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

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

  • 如果一个呼叫已经在执行(说到api/books/1337或来api/editors),应该通知呼叫者它正在加载数据,并且应该给它一个旧的(如果它存在),就像它是原始呼叫者一样.

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

问题:

在观看和研究Virgil Dobjanschi在Google I/O 2010上提出的所有组件之后,我对此表示怀疑:

  1. 如何透明地为任何呼叫者处理" 实体更新 "概念?我应该使用ContentObserverContentProvider我将要实施?

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

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

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

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

jcw*_*ger 10

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

  1. 一个应用程序 它使用一组Activity沿 CursorContentObserver,也许CursorAdapter有的提供UI从本地存储的数据ContentProvider.
  2. ContentProvider 本地设备的数据存储.处理CRUD调用,处理通知SyncAdapter需要将更新推送到服务器.
  3. Account 远程服务器上的用户标识.
  4. SyncAdapter 后台进程,它运行并使本地数据存储与服务器保持同步.
  5. 服务器本身.

所以.对于问题:

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

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

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

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

  5. 一个.你可以通过调用强制同步ContentResolver.RequestSync()ContentResolver.SYNC_EXTRAS_MANUAL的额外软件包.
    您可以使用客户端代码手动获取内容并直接将其推送到ContentProvider中.

  • ContentProvider永远不会获取更新的数据.它只是您本地的存储提供商.你有两种选择.1.您可以在ContentProvider中为"请求更新"构建一个表,然后让SyncAdapter读取该表以强制拉取,然后一旦有结果,将它们推送到ContentProvider.第二个选项,您的应用程序本身会生成一个后台线程,获取数据,然后推送到ContentProvider.单独构建数据提取和解析类,并使SyncAdapter和应用程序调用它们.无论哪种方式,您都可以将行标记为"更新". (4认同)