编辑:我已经修改了这个答案,以包括我最初称为“编写专门的 ContentProvider”的方法示例代码。这应该完全满足问题的要求。可能答案太大了,但它现在有内部代码依赖关系,所以让我们把它作为一个整体。要点仍然成立:如果您愿意,请使用下面的 ContentPrvder,但尝试将file://
Uris 提供给支持它们的应用程序,除非您想因某人的应用程序崩溃而受到指责。
原始答案
我会像现在一样远离存储访问框架。它没有得到谷歌的充分支持,应用程序的支持也很糟糕,很难区分这些应用程序中的错误和 SAF 本身。如果您有足够的信心(这实际上意味着“可以比一般的 Android 开发人员更好地使用 try-catch 块”),请自己使用存储访问框架,但只将好的旧file://
路径传递给其他人。
您可以使用以下技巧从 ParcelFileDescriptor 获取文件系统路径(您可以通过调用openFileDescriptor从 ContentResolver 获取它):
class FdCompat {
public static String getFdPath(ParcelFileDescriptor fd) {
final String resolved;
try {
final File procfsFdFile = new File("/proc/self/fd/" + fd.getFd());
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
// Returned name may be empty or "pipe:", "socket:", "(deleted)" etc.
resolved = Os.readlink(procfsFdFile.getAbsolutePath());
} else {
// Returned name is usually valid or empty, but may start from
// funny prefix if the file does not have a name
resolved = procfsFdFile.getCanonicalPath();
}
if (TextUtils.isEmpty(resolved) || resolved.charAt(0) != '/'
|| resolved.startsWith("/proc/") || resolved.startsWith("/fd/"))
return null;
} catch (IOException ioe) {
// This exception means, that given file DID have some name, but it is
// too long, some of symlinks in the path were broken or, most
// likely, one of it's directories is inaccessible for reading.
// Either way, it is almost certainly not a pipe.
return "";
} catch (Exception errnoe) {
// Actually ErrnoException, but base type avoids VerifyError on old versions
// This exception should be VERY rare and means, that the descriptor
// was made unavailable by some Unix magic.
return null;
}
return resolved;
}
}
您必须做好准备,上面的方法将返回 null(文件是管道或套接字,这是完全合法的)或空路径(对文件的父目录没有读取权限)。如果发生这种情况,请将整个流复制到您可以访问的某个目录。
完整的解决方案
如果您真的想坚持使用内容提供商 Uris,那就去吧。以下面的 ContentProvider 的代码为例。粘贴到您的应用中(并在 AndroidManifest 中注册)。使用getShareableUri
以下方法将收到的 Storage Access Framework Uri 转换为您自己的。将该 Uri 传递给其他应用程序,而不是原始 Uri。
下面的代码是不安全的(您可以轻松地使其安全,但解释这会使这个答案的长度超出想象)。如果您愿意,请使用file://
Uris——Linux 文件系统被广泛认为足够安全。
扩展下面的解决方案以提供没有相应 Uri 的任意文件描述符作为练习留给读者。
public class FdProvider extends ContentProvider {
private static final String ORIGINAL_URI = "o";
private static final String FD = "fd";
private static final String PATH = "p";
private static final Uri BASE_URI =
Uri.parse("content://com.example.fdhelper/");
// Create an Uri from some other Uri and (optionally) corresponding
// file descriptor (if you don't plan to close it until your process is dead).
public static Uri getShareableUri(@Nullable ParcelFileDescriptor fd,
Uri trueUri) {
String path = fd == null ? null : FdCompat.getFdPath(fd);
String uri = trueUri.toString();
Uri.Builder builder = BASE_URI.buildUpon();
if (!TextUtils.isEmpty(uri))
builder.appendQueryParameter(ORIGINAL_URI, uri);
if (fd != null && !TextUtils.isEmpty(path))
builder.appendQueryParameter(FD, String.valueOf(fd.getFd()))
.appendQueryParameter(PATH, path);
return builder.build();
}
public boolean onCreate() { return true; }
public ParcelFileDescriptor openFile(Uri uri, String mode)
throws FileNotFoundException {
String o = uri.getQueryParameter(ORIGINAL_URI);
String fd = uri.getQueryParameter(FD);
String path = uri.getQueryParameter(PATH);
if (TextUtils.isEmpty(o)) return null;
// offer the descriptor directly, if our process still has it
try {
if (!TextUtils.isEmpty(fd) && !TextUtils.isEmpty(path)) {
int intFd = Integer.parseInt(fd);
ParcelFileDescriptor desc = ParcelFileDescriptor.fromFd(intFd);
if (intFd >= 0 && path.equals(FdCompat.getFdPath(desc))) {
return desc;
}
}
} catch (RuntimeException | IOException ignore) {}
// otherwise just forward the call
try {
Uri trueUri = Uri.parse(o);
return getContext().getContentResolver()
.openFileDescriptor(trueUri, mode);
}
catch (RuntimeException ignore) {}
throw new FileNotFoundException();
}
// all other calls are forwarded the same way as above
public Cursor query(Uri uri, String[] projection, String selection,
String[] selectionArgs, String sortOrder) {
String o = uri.getQueryParameter(ORIGINAL_URI);
if (TextUtils.isEmpty(o)) return null;
try {
Uri trueUri = Uri.parse(o);
return getContext().getContentResolver().query(trueUri, projection,
selection, selectionArgs, sortOrder);
} catch (RuntimeException ignore) {}
return null;
}
public String getType(Uri uri) {
String o = uri.getQueryParameter(ORIGINAL_URI);
if (TextUtils.isEmpty(o)) return "*/*";
try {
Uri trueUri = Uri.parse(o);
return getContext().getContentResolver().getType(trueUri);
} catch (RuntimeException e) { return null; }
}
public Uri insert(Uri uri, ContentValues values) {
return null;
}
public int delete(Uri uri, String selection, String[] selectionArgs) {
return 0;
}
public int update(Uri uri, ContentValues values, String selection,
String[] selectionArgs) { return 0; }
}