【问题标题】:Google Drive Android API: Deleted folder still exists in queryGoogle Drive Android API:查询中仍然存在已删除的文件夹
【发布时间】:2015-09-11 19:29:27
【问题描述】:

运行下面的代码,我在平板电脑上使用 Google Drive Android API 创建了一个文件夹。几秒钟后,从 PC 上的远程位置删除该文件夹。当我重新运行代码时,API 仍然认为“MyFolder”存在,即使它已被删除并且在平板电脑上的 Google Drive 应用程序中不可见。文件夹持久性最终在一段时间后消失,代码按预期工作。这是云驱动器的预期行为吗?

Query query = new Query.Builder()
        .addFilter(Filters.and(Filters.eq(
                SearchableField.TITLE, "MyFolder"),
                Filters.eq(SearchableField.TRASHED, false)))
        .build();
Drive.DriveApi.query(getGoogleApiClient(), query)
        .setResultCallback(new ResultCallback<DriveApi.MetadataBufferResult>() {
    @Override
    public void onResult(DriveApi.MetadataBufferResult result) {
        if (!result.getStatus().isSuccess()) {
            showMessage("Cannot create folder in the root.");
        } else {
            boolean isFound = false;
            for(Metadata m : result.getMetadataBuffer()) {
                if(!isFound) {
                    if (m.getTitle().equals("MyFolder")) {
                        showMessage("Folder exists");
                        isFound = true;
                    }
                }
            }
            if(!isFound) {
                showMessage("Folder not found; creating it.");
                MetadataChangeSet changeSet = new MetadataChangeSet.Builder()
                        .setTitle("MyFolder")
                        .build();
                Drive.DriveApi.getRootFolder(getGoogleApiClient())
                        .createFolder(getGoogleApiClient(), changeSet)
                        .setResultCallback(new ResultCallback<DriveFolder.DriveFolderResult>() {
                    @Override
                    public void onResult(DriveFolder.DriveFolderResult result) {
                        if (!result.getStatus().isSuccess()) {
                            showMessage("Error while trying to create the folder");
                        } else {
                            mThwingAlbertFolderId = result.getDriveFolder().getDriveId();
                            showMessage("Created a folder: " + mThwingAlbertFolderId);
                        }
                    }
                });
            }
        }
    }
});

【问题讨论】:

  • 仅供参考,它与您的问题没有直接关系,但是您应该更改“isFound”逻辑,而不是在for(Metadata m...) 中使用if,一旦找到您的文件夹,您应该使用break 命令以获得更好的性能。
  • 是的。我打算添加逻辑来处理重复文件夹名称的多个实例,但由于这个问题而从未完成。谢谢。

标签: android google-drive-android-api


【解决方案1】:

您所看到的是GDAA 的“正常”行为,如果您仔细查看“Lifecycle of a Drive file”图表可以解释这一点(警告:我从未见过源代码,只是假设根据我的观察)。

请看,GDAA 与REST Api 不同,它创建了一个层来尽最大努力创建缓存和网络流量优化。因此,当您从“外部”(如 Web 应用程序)操作文件/文件夹时,GDAA 层在启动同步之前不知道这一事实,由它自己的逻辑控制。我自己最初认为 GooDrive 通过向 GDAA 发送某种通知来控制这种情况,但显然情况并非如此。此外,一些谷歌员工提到了'requestSync()' as a cure,但我从未成功过。

你认为你在做的是轮询 GooDrive。但实际上,您正在轮询其 DriveId 仍然有效(未更新)的 GDAA(本地 GooPlaySvcs),这与已经消失的真正 GooDrive 对象不同。

这是文档中没有明确说明的一件事。 GDAA 不是最好的 Api for EVERY application。它的缓存机制非常适合透明地管理在线/离线状态、网络流量优化。电池寿命,...但是在您的情况下,可能使用 REST Api 会更好,因为您获得的响应反映了当前的 GooDrive 状态。

我自己也遇到过类似的情况,不得不从 GDAA 切换回 REST(并用基于私有 GCM 的通知系统代替轮询)。不用说,通过使用 REST Api,您的应用程序会变得更加复杂,通常需要同步适配器/服务来进行数据同步、管理网络状态……所有 GDAA 都免费提供给您)。
如果您想同时使用 2 个 api,可以在 Github 上使用两个相同的 CRUD 实现(GDAAREST)。

祝你好运

【讨论】:

  • 谢谢你,@seanpj。完美答案!我开始使用 REST 并看到了潜力,但我认为最好使用 Android API,因为这是 Google 开发人员页面告诉我的。我会换回来的。
  • ...但要注意管理延迟响应、开/离线状态、..所有不符合 UI 线程流的事情的痛苦 :-)。
【解决方案2】:

Google drive api 不会立即同步,这就是为什么删除的文件夹仍然显示的原因,所以你必须使用 requestSync() 强制 google drive 同步

Drive.DriveApi.requestSync(mGoogleApiClient).await();

我在这里找到了一个示例 sn-p: http://wiki.workassis.com/android-google-drive-api-deleted-folder-still-exists-in-query/

【讨论】:

  • 你拯救了我的一天......谢谢很多
【解决方案3】:

正如 Sean 所提到的,Drive Android API 会在本地缓存元数据以减少带宽和电池使用量。

当您在设备上执行操作时,例如创建文件夹时,我们会尝试尽快在服务器上应用该操作。尽管由于动作依赖和内容传输可能会出现延迟,但您通常会很快看到结果反映在服务器上。

在服务器上执行操作时,例如通过网络客户端删除文件夹,此操作会在下次 Drive Android API 同步时反映在设备上。为了节省电池和带宽,同步频率取决于 API 的使用方式,因为这是用户的优先事项。

如果您需要保证同步已经发生,您可以使用DriveApi.requestSync() 明确请求同步并等待结果。目前,此速率限制为每分钟 1 次,在测试期间经常遇到这种情况,但对实际使用的影响应该小得多。

如果此同步行为导致您的用例出现问题,请通过 issue tracker 告知我们,以便我们调查解决方案。

【讨论】:

  • 感谢您的回复。
【解决方案4】:

Google Drive 使用自己的 Drive api 生命周期并管理缓存中的所有内容,这就是为什么如果您删除某些文件或文件夹并尝试使用 google drive api 访问它仍然可用,因为它存储在缓存中,因此您需要明确为此调用requestSync() 方法,然后在该缓存将被更新并为您提供找不到的文件夹或文件。 下面是代码:

Drive.DriveApi.requestSync(mGoogleApiClient).setResultCallback(new ResultCallback<Status>() {
        @Override
        public void onResult(@NonNull Status status) {
            Log.e("sync_status", status.toString());
            if (status.getStatus().isSuccess()) {
                setRootFolderDriveId();
            }
        }
    });

不要调用Drive.DriveApi.requestSync(mGoogleApiClient).await(),因为你的主线程会阻塞,所以它会崩溃。使用上面的一个,在成功回调后你可以在谷歌驱动器上进行操作,因为它已经更新了。

【讨论】:

    【解决方案5】:

    你可以在主线程中做:

     Drive.DriveApi.requestSync(mGoogleApiClient).setResultCallback(new ResultCallback<com.google.android.gms.common.api.Status>() {
        @Override
        public void onResult(com.google.android.gms.common.api.Status status) {
            if (!status.getStatus().isSuccess()) {
                Log.e("SYNCING", "ERROR" + status.getStatusMessage());
            } else {
                Log.e("SYNCING", "SUCCESS");
                // execute your code to interact with Google Drive
    
            }
        }
    });
    

    【讨论】:

      【解决方案6】:

      我遇到了同样的问题,使用“Drive.DriveApi.requestSync”就可以了。 另外我建议看一下https://github.com/francescocervone/RxDrive,因为您可以使用 rxandroid 将同步连接到其他驱动器操作。 例如,这变成了一个删除和同步操作:

      Observable<Boolean> deleteFile = rxDrive.delete(file);
      Observable<Void> syncDrive = rxDrive.sync();
      Observable.concat(deleteFile, syncDrive);
      

      【讨论】:

        【解决方案7】:

        您从查询中列出已删除文件的原因是 Google 云端硬盘有一个“可搜索”的“垃圾箱”文件夹。您需要先清空垃圾箱。

        【讨论】:

          猜你喜欢
          • 1970-01-01
          • 2014-03-09
          • 1970-01-01
          • 2020-08-14
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          相关资源
          最近更新 更多