如何使用新的文件提供程序共享当前应用的 APK(或其他已安装的应用)?



Background

过去,使用一个简单的命令可以轻松地与您想要的任何应用程序共享APK文件:

startActivity(new Intent(Intent.ACTION_SEND,Uri.fromFile(filePath)).setType("*/*"));

问题所在

如果您的应用面向Android API 24(Android Nougat)或更高版本,则上述代码将导致崩溃,由FileUriExposedException引起(在这里写过,打开APK文件的示例解决方案可以在这里找到)。

这实际上对我很好,使用以下代码:

File apkFile = new File(apkFilePathFromSomewhereInExternalStorage);
Intent intent = new Intent(Intent.ACTION_SEND);
Uri fileUri = FileProvider.getUriForFile(this, getApplicationContext().getPackageName() + ".provider", apkFile);
intent.setType("text/plain");
intent.putExtra(Intent.EXTRA_STREAM, fileUri);
intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
startActivity(intent);

但是,问题是我也希望能够共享当前应用程序的APK(以及其他已安装的应用程序)。

为了获取当前应用的APK的路径,我们使用这个:

final PackageInfo packageInfo = getPackageManager().getPackageInfo(getPackageName(), 0);
File apkFile=new File(packageInfo.applicationInfo.publicSourceDir);
...

所有应用程序都可以访问此APK,而无需任何许可,每个已安装应用程序的APK也是如此。

但是当我将此文件与上述代码一起使用以使用文件提供程序进行共享时,我得到以下异常:

IllegalArgumentException: Failed to find configured root that contains ...

当我使用符号链接到APK的文件时也是如此,如下所示:

File apkFile=new File(packageInfo.applicationInfo.publicSourceDir);
final String fileName = "symlink.apk";
File symLinkFile = new File(getFilesDir(), fileName);
if (!symLinkFile.exists())
symLinkPath = symLinkFile.getAbsolutePath();
createSymLink(symLinkPath, apkFile.getAbsolutePath());
Uri fileUri= FileProvider.getUriForFile(this, getApplicationContext().getPackageName() + ".provider", symLinkFile);
...
public static boolean createSymLink(String symLinkFilePath, String originalFilePath) {
try {
if (VERSION.SDK_INT >= VERSION_CODES.LOLLIPOP) {
Os.symlink(originalFilePath, symLinkFilePath);
return true;
}
final Class<?> libcore = Class.forName("libcore.io.Libcore");
final java.lang.reflect.Field fOs = libcore.getDeclaredField("os");
fOs.setAccessible(true);
final Object os = fOs.get(null);
final java.lang.reflect.Method method = os.getClass().getMethod("symlink", String.class, String.class);
method.invoke(os, originalFilePath, symLinkFilePath);
return true;
} catch (Exception e) {
e.printStackTrace();
}
return false;
}

我尝试过什么

我尝试使用我认为会有所帮助的各种组合来配置provider_paths.xml文件,例如其中任何一个:

<external-path name="external_files" path="."/>
<external-path path="Android/data/lb.com.myapplication/" name="files_root" />
<external-path path="." name="external_storage_root" />
<files-path name="files" path="." />
<files-path name="files" path="" />

我还尝试禁用与此机制关联的 strictMode:

StrictMode.VmPolicy.Builder builder = new StrictMode.VmPolicy.Builder();
StrictMode.setVmPolicy(builder.build());

问题

如何从我的应用程序可以访问的每个可能路径共享任何APK文件,包括使用符号链接文件?

但是当我将此文件与上述代码一起使用以使用文件提供程序进行共享时,我得到此异常

那是因为packageInfo.applicationInfo.publicSourceDir不在FileProvider的可能根源之一之下。

当我使用符号链接到APK的文件时也是如此

也许符号链接在这里不起作用。您的<files-path>元素之一(或完全关闭path元素)可以在getFilesDir()之外提供普通文件。

如何从我的应用程序可以访问的每个可能路径共享任何APK文件,包括使用符号链接文件?

由于没有对符号链接的官方支持,我无法在那里帮助你。

否则,您有三个主要选项:

  1. FileProvider喜欢的地方复制APK文件

  2. 尝试将我的StreamProvider与一些自定义扩展一起使用,以教它从packageInfo.applicationInfo.publicSourceDir

  3. 编写您自己的ContentProvider,从packageInfo.applicationInfo.publicSourceDir

我记得,apk 文件位于/data/app/your.package.name-1(或 -2)/base.apk 自 SDK 21(或 23?如果我错了,请纠正我。在此之前.apk/data/app/your.package.name。所以试试这个:

<root-path path="data/app/" name="your.name">

我在这里的回答不适用于您的情况。我在那里的评论部分解释了这一点。

PS:正如@CommonsWare所说,此功能未记录在案。将来有可能将其删除。所以使用这个是有风险的。

相关内容

  • 没有找到相关文章

最新更新