使用Intent安装APK方法(兼容Android N)

Android N之前,通过Intent安装一个APK代码差多不是下面这个样子的:

Intent install = new Intent(Intent.ACTION_VIEW);
install.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
File apkFile = new File(getExternalFilesDir(Environment.DIRECTORY_DOWNLOADS),"app.apk";
install.setDataAndType(Uri.fromFile(apkFile)), "application/vnd.android.package-archive");
startActivity(install)

上面代码在Android N(API 26)之前是没有问题的,但是从Android 7.0开始,系统修改了安全机制: 限定应用在默认情况下只能访问自身应用数据。所以当我们想通过File对象访问其它package数据时,就需要借助于ContentProvider、FileProvider这些组件,否则会报FileUriExposedException异常。

相应的,显然上面的代码在Android N中会有FileUriExposedException的问题,所以我们需要借助FileProvider做出下面的修改。

使用FileProvider的步骤
添加Provider到AndroidManifest.xml

<provider
      android:name="android.support.v4.content.FileProvider"
      android:authorities="${applicationId}.android7.fileprovider"
      android:exported="false"
      android:grantUriPermissions="true">
      <meta-data
        android:name="android.support.FILE_PROVIDER_PATHS"
        android:resource="@xml/file_paths" />
</provider>

创建@xml/file_paths对应的资源文件

<?xml version="1.0" encoding="utf-8"?>
<paths>

    <!-- 代表外部存储区域的根目录下的文件 Environment.getExternalStorageDirectory() 根目录 -->
    <!-- for: storage/emulated/0/ -->
    <external-path
        name="external_storage_root"
        path="." />  

    <!-- 代表外部存储区域的根目录下的文件 Environment.getExternalStorageDirectory()/Pictures/ 根目录 -->
    <!-- for: storage/emulated/0/Pictures/ -->
    <external-path
        name="Pictures"
        path="Pictures/" />

    <!-- 代表app 私有的存储区域 Context.getCacheDir() 根目录下的images目录 -->
    <!-- for: /data/user/0/com.hm.demo/cache/images -->
    <cache-path
        name="hm_private_cache"
        path="images" />

    <!-- 代表app 私有的存储区域 Context.getFilesDir() 根目录下的images目录 -->
    <!-- for: /data/user/0/com.hm.demo/files/images -->
    <files-path
        name="hm_private_files"
        path="images" />

    <!-- 代表app 外部存储区域根目录下的文件 Context.getExternalCacheDir 根目录 -->
    <!-- for: /storage/emulated/0/Android/data/com.hm.demo/cache/-->
    <external-cache-path
        name="external_cache"
        path="." />

    <!-- 代表app 外部存储区域根目录下的文件 Context.getExternalFilesDir 根目录 -->
    <!-- for: /storage/emulated/0/Android/data/com.hm.demo/files/-->
    <external-files-path
        name="external_files"
        path="." />

</paths>

paths里面用来放你需要访问的非本应用路径,只有通过这里申明后,才能在程序中使用不报错。

  • files-path 对应 Context.getFilesDir()
  • cache-path 对应 getCacheDir()
  • external-path 对应 Environment.getExternalStorageDirectory()
  • external-files-path 对应 Context#getExternalFilesDir(String) Context.getExternalFilesDir(null)
  • external-cache-path 对应 Context.getExternalCacheDir()

具体其中的path和name就需要你自己来根据需求确定了。

应用代码

Intent install = new Intent(Intent.ACTION_VIEW);
install.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
File apkFile = new File(getExternalFilesDir(Environment.DIRECTORY_DOWNLOADS),"app.apk";

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
    install.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
    Uri contentUri = FileProvider.getUriForFile(context, BuildConfig.APPLICATION_ID+".android7.fileProvider", apkFile);
    install.setDataAndType(contentUri, "application/vnd.android.package-archive");
} else {
    install.setDataAndType(Uri.fromFile(apkFile), "application/vnd.android.package-archive");
}

startActivity(install);

Google官方链接


如有误点,多多指教 喜欢呦~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。