3

背景

过去,使用简单的命令可以很容易地与您想要的任何应用程序共享 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 也是如此。

但是,当我将此文件与上述代码一起使用 FileProvider 进行共享时,我得到了这个异常:

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 文件,包括使用符号链接文件?

4

2 回答 2

1

但是,当我将此文件与上述代码一起使用 FileProvider 进行共享时,我得到了这个异常

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

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

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

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

由于没有对符号链接的官方支持,因此我无法为您提供帮助。

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

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

  2. 尝试使用myStreamProvider和一些自定义扩展来教它从packageInfo.applicationInfo.publicSourceDir

  3. 写你自己的ContentProvider,服务于packageInfo.applicationInfo.publicSourceDir

于 2016-12-02T22:30:52.817 回答
0

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

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

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

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

于 2016-12-04T07:15:28.297 回答