Android:如何在同一用户的设备之间同步存储在 Room 数据库中的应用程序数据?

Bha*_*rat 6 android cloudkit android-room google-cloud-firestore

我正在开发一个在本地存储用户数据的应用程序。我已经使用Room数据库将用户创建的数据存储在 Android 设备中。现在,如果用户拥有不止一台设备,那么数据应该可以在他们的所有 (Android) 设备上使用,而无需任何中央数据库或 Web 服务。

注意:对于 iOS 端,我曾经CloudKit在 iOS 设备之间同步数据。有没有办法对 Android 设备做同样的事情?请注意,我不想将数据存储在任何中央数据库或网络服务器或任何中央云存储上。它应该只归用户所有。(即它应该通过用户的谷歌帐户同步)。 注意:我想要一个原生的 android 解决方案,不需要跨平台(Android 到 iOS)同步。

更新:到目前为止,我发现了一些选项,如Firestore实时数据库,一些 AWS 服务也提供同步工具,但所有这些的问题在于它们都将用户数据存储在中央云位置。

所以我的问题仍然是一样的 - 有没有办法在使用Google Drive或类似的东西的android 设备之间同步数据,我们可以利用用户自己的云存储,而不是将用户数据存储在中央数据库中。

Pav*_*sha 8

编辑

没有本地/默认/开箱机制来同步数据,实时的Android

不过也有代言人。

在 Android 的上下文中,出现了两个主要选项:

  • 具有一些持久存储的套接字 - 您可以实现支持套接字的后端服务器,并为您的 android(ios/web/desktop) 应用程序添加此支持。这是在设备之间实现实时同步的最常见和非常低级的机制。优点:跨平台通用、行业标准、大量在线信息。缺点:因为它是相对底层的东西,需要很多代码,如果有很多同步事件,需要后端,需要单独的持久性,可能会非常复杂和复杂。主要实现:Socket.io

  • Firebase实时数据库/ Cloud Firestore - 基本上是将套接字与某些数据库结合起来进行包装。“优点”:既不需要后端也不需要数据库,原型/低负载应用程序免费,非常直接的实现,跨平台,效果很好。缺点:如果你想走“干净的方式”,有很多样板,你无法控制内部流程,在高负载时不是很明显的支付计算,你将自己与具有特定 API 的第三方后端联系起来 - 需要明确定义的架构如果您想在没有完整的应用程序重制的情况下从它移到其他地方。比较:Firebase Cloud Firestore 与实时数据库

我不会详细介绍实现,因为在这种情况下官方文档非常简单(与谷歌驱动器不同)

Cloud Firestore 文件

Realtime Database 文件

编辑结束

是的。有可能使用Google Drive. 它只能由您的应用程序访问。此外,它不会占用用户的任何空间,Google Drive这非常方便。在这种情况下跨平台是可能的,但它需要为 iOs 和 Android 单独实现(SDK 不支持开箱即用)。

因此,该解决方案基于Google Drive API,更具体地说是基于Store 应用程序特定的数据部分。

与此相关的用户操作唯一可见的是 google 登录提示。

是 Java 的详细快速入门

因此,为了做到这一点,您需要:

  • 添加依赖项;

  • 这里这里描述的那样实现谷歌身份验证

  • 非常重要使用drive.appdatadrive.file范围对用户进行身份验证。还要在您的 Google Cloud Console 中启用Google Drive API。所有这些都在这里描述

  • 找出您的Room SQLite文件所在的位置,如下所示:String currentDBPath = context.getDatabasePath("your_database_name.db").getAbsolutePath();

  • 这里描述的那样将文件上传到 Google Drive App Folder

基本上它看起来像这样

  • 在控制台中启用Google Dive API

  • 添加依赖(可能需要更新到相关版本)

implementation 'com.google.android.gms:play-services-auth:17.0.0'// for google sign in

// for drive integration
implementation 'com.google.android.gms:play-services-auth:16.0.1'
implementation 'com.google.http-client:google-http-client-gson:1.26.0'
implementation('com.google.api-client:google-api-client-android:1.26.0') {
exclude group: 'org.apache.httpcomponents'
}
implementation('com.google.apis:google-api-services-drive:v3-rev136-1.25.0') 
{
exclude group: 'org.apache.httpcomponents'
} 
Run Code Online (Sandbox Code Playgroud)
  • androidgradle 标签中
packagingOptions {
exclude 'META-INF/DEPENDENCIES'
exclude 'META-INF/LICENSE'
exclude 'META-INF/LICENSE.txt'
exclude 'META-INF/license.txt'
exclude 'META-INF/NOTICE'
exclude 'META-INF/NOTICE.txt'
exclude 'META-INF/notice.txt'
exclude 'META-INF/ASL2.0'
}
Run Code Online (Sandbox Code Playgroud)
  • 显现
<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
Run Code Online (Sandbox Code Playgroud)
  • 具有所需范围的 Google 身份验证
       val gso = GoogleSignInOptions.Builder(GoogleSignInOptions.DEFAULT_SIGN_IN)
            .requestScopes(
                 Scope(Scopes.DRIVE_FILE),
                 Scope(Scopes.DRIVE_APPFOLDER)
            )
            .requestEmail()
            .build()
        googleSingInClient = GoogleSignIn.getClient(this, gso)
        val signInIntent = googleSingInClient.signInIntent
        startActivityForResult(signInIntent, RC_SIGN_IN)
        ... // handle sign in result
Run Code Online (Sandbox Code Playgroud)
  • 设置谷歌驱动客户端服务(上下文所在)
        val mAccount = GoogleSignIn.getLastSignedInAccount(this)

        val credential =
            GoogleAccountCredential.usingOAuth2(
                applicationContext, Collections.singleton(Scopes.DRIVE_FILE)
            )
        credential.setSelectedAccount(mAccount.getAccount())
        val googleDriveService =
            Drive.Builder(
                AndroidHttp.newCompatibleTransport(),
                GsonFactory(),
                credential
            )
                .setApplicationName("Your app name")
                .build()
Run Code Online (Sandbox Code Playgroud)
  • 创建和发送文件
        val fileMetadata = File()
        fileMetadata.setName("your_database_name.db")
        fileMetadata.setParents(Collections.singletonList("appDataFolder"))
        val databasePath = context.getDatabasePath("your_database_name.db").absolutePath
        val filePath = File(databasePath)
        val mediaContent = FileContent("application/sqlite", filePath)
        val file = driveService.files().create(fileMetadata, mediaContent)
            .setFields("id")
            .execute()
Run Code Online (Sandbox Code Playgroud)
  • 在其他设备上搜索文件
        val files = driveService.files().list()
            .setSpaces("appDataFolder")
            .setFields("nextPageToken, files(id, name)")
            .setPageSize(10)
            .execute()
        for (file in files.getFiles()) {
            // get file here 
        }
Run Code Online (Sandbox Code Playgroud)

您可能会在此处获得更多信息

希望能帮助到你。