6

我正在尝试使用 DocumentFile (由于存储访问框架)在创建文件之前检查文件是否存在。但是DocumentFile().fromTreeUri()删除了可能的 Uri 中不存在的部分,这将导致DocumentFile().exists()始终返回 true,而不管它是否存在。

我创建了一个简单的例子来证明我的观点。首先我们要求用户选择一个目录:

@Override
protected void onCreate(Bundle savedInstanceState)
{
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    
    // Ask the user for the source folder
    Intent intent = new Intent(Intent.ACTION_OPEN_DOCUMENT_TREE);
    startActivityForResult(intent, 100);
}

在响应时,我们添加/fictionalFile到路径(从而使其成为不存在的文件),然后检查它是否存在:

public void onActivityResult(int requestCode, int resultCode, Intent resultData)
{
    if (resultCode == RESULT_OK)
    {
        if(requestCode == 100)
        {
            Uri fictionalURI = Uri.parse(resultData.getData()+"/fictionalFile");
            DocumentFile fictionalFile = DocumentFile.fromTreeUri(this, fictionalURI);
            Log.i("STORAGE", "FICTIONAL URI: "+fictionalURI);
            Log.i("STORAGE", "FICTIONAL DOCUMENTFILE URI: "+fictionalFile.getUri());

            if(fictionalFile.exists())
            {
                Log.i("STORAGE", "Fictional file exists");
            }
        }
    }
}

但是,当 DocumentFile.fromTreeUri() 在虚构的 Uri 上运行时,虚假的“/fictionalfile”部分会丢失,这会导致 DocumentFile.exists() 函数返回 true,如下面的 LogCat 所示:

I/STORAGE:虚构 URI:content://com.android.externalstorage.documents/tree/17FA-1C18%3AFileSync%2Ftarget/fictionalFile

I/STORAGE: FICTIONAL DOCUMENTFILE URI: content://com.android.externalstorage.documents/tree/17FA-1C18%3AFileSync%2Ftarget/document/17FA-1C18%3AFileSync%2Ftarget

I/STORAGE:虚构文件存在

(在上面的例子中,我使用的是 SD 卡,因此路径名很长)

是否有另一种方法来检查尚未创建的 DocumentFile 是否存在?用例是,将文件从目录 A 复制到目录 B 时,我想在开始传输之前检查目录 B 中是否已经存在所述文件。

更新:我现在意识到 usingDocumentFile.fromTreeUri()是错误的,我应该使用DocumentFile.fromSingleUri(). 这有帮助,但是在.exists()新文件上运行时,我得到W/DocumentFile: Failed query: java.lang.UnsupportedOperationException: Unsupported Uri content://com.android.externalstorage.documents/tree/17FA-1C18%3AFileSync%2Ftarget/fictionalFile. 有什么想法吗?

public void onActivityResult(int requestCode, int resultCode, Intent resultData)
{
    if (resultCode == RESULT_OK)
    {
        if(requestCode == 100)
        {
            Uri fictionalURI = Uri.parse(resultData.getData()+"/fictionalFile");
            DocumentFile fictionalFile = DocumentFile.fromSingleUri(this, fictionalURI);
            Log.i("STORAGE", "FICTIONAL URI: "+fictionalURI);
            Log.i("STORAGE", "FICTIONAL DOCUMENTFILE URI: "+fictionalFile.getUri());

            if(fictionalFile != null && fictionalFile.exists())
            {
                Log.i("STORAGE", "Fictional file exists");
            }
        }
    }
}
4

2 回答 2

6

给定treeUri作为由Uri返回的ACTION_OPEN_DOCUMENT_TREE,包装treeUri在一个DocumentFileusing 中fromTreeUri(),然后调用findFile()提供DocumentFile您正在寻找的显示名称的那个(例如,fictionalFile)。如果返回null,则没有与该显示名称匹配的文件。

爱荷华州:

if (DocumentFile.fromTreeUri(this, treeUri).findFile(whatevs) == null) {
  // TODO: something
}

但请注意,“显示名称”不一定是文件名。

于 2018-08-02T21:55:34.690 回答
1

就我而言,当我使用 applicationContext 签DocumentFile.fromSingleUri(context, media.fileUri)?.exists() == true入 CoroutineWorker 时,当从外部应用程序(例如,在文件管理器中)删除文件时,DocumentProvider 总是返回 true。

问题已解决uri.isFileExist(context)

import android.content.Context
import android.net.Uri
import androidx.documentfile.provider.DocumentFile

fun DocumentFile?.isFile() = this?.isFile ?: false

fun DocumentFile?.isExists() = this?.exists() ?: false

fun DocumentFile?.getLength() = this?.length() ?: 0

fun DocumentFile?.isFileExist() = isFile() && isExists() && getLength() > 0

fun Uri.isFileExist(context: Context) =
    DocumentFile.fromSingleUri(context, this).isFileExist()
于 2020-08-20T14:50:00.563 回答