在Android上同步SQL数据库与REST远程服务器的最佳实践

时间:2012-03-05 10:53:37

标签: android sql sqlite synchronization android-contentprovider

我有一个自定义Android ContentProvider,用于存储和检索SQLite数据库中的数据。

我们假设其中一个数据库表有一个_ID列和一个NAME列以及如下内容:

|==========|==========|
|   _ID    |   NAME   |
|==========|==========|
|    1     |    d1    |
|    2     |    d2    |
|    3     |    d3    |
|    4     |    d4    |
|==========|==========|

此SQLite数据库与远程数据库保持同步,并通过网络定期获取新数据。 表格上的可能操作如下:

  1. 现有行可以删除
  2. 可以添加新行
  3. 可以修改现有行的NAME列
  4. 现在假设所有可能的操作立即发生,并且在从远程服务器获取一些最新数据之后,该表的新内容必须设置如下:

    |==========|==========|
    |   _ID    |   NAME   |
    |==========|==========|
    |    1     |    d7    |
    |    3     |    d3    |
    |    4     |    d6    |
    |    5     |    d5    |
    |==========|==========|
    

    可以采用两种不同的方法:

    1. 查询数据库并检查每个现有行以查看是否需要更新,然后添加任何新行并删除任何缺失的行 - 尽管如果我们想要使用分页更新数据库,这种方法可能有点棘手方法而不是单一网络提取
    2. 使用单个DELETE SQL命令删除整个表,然后添加从服务器收到的所有行
    3. 在Android上,我目前正在使用批处理操作实现第二种方法,以最大限度地提高性能:

      final ArrayList<ContentProviderOperation> operations = new ArrayList<ContentProviderOperation>();
      
      // with this URI, the content provider deletes all rows
      operations.add(ContentProviderOperation.newDelete(Users.CONTENT_URI).build());
      
      final ContentValues values = new ContentValues();
      values.put(ID_COLUMN, 1);
      values.put(NAME_COLUMN, "d7");
      values.put(ID_COLUMN, 3);
      values.put(NAME_COLUMN, "d3");
      values.put(ID_COLUMN, 4);
      values.put(NAME_COLUMN, "d6");
      values.put(ID_COLUMN, 5);
      values.put(NAME_COLUMN, "d5");
      
      operations.add(ContentProviderOperation.newInsert(Users.CONTENT_URI).withValues(values).build());
      
      getApplicationContext().getContentResolver().applyBatch(MyContentProvider.AUTHORITY, operations);
      

      这是最好的方法,还是方法1(或其他方法)在性能方面更好?

      编辑:例如,采用方法2,使用数据库事务的重写ContentProvider#bulkInsert可以大大加快批量插入操作:请参阅this question

1 个答案:

答案 0 :(得分:1)

最佳选择需要正确的API实现 - 何时应存储一些db_version(上次更新的int或时间戳,或其他)。

在更新服务器期间响应数据和操作类型 - 添加,更新,删除。