android7.0更新安装apk的方法

StrictMode模式

从android7.0开始强制启用StrictMode“严苛模式”。StrictMode是在android2.3引进的类。当时它的作用是作为一个开发工具用的,开发者可以开发者选项中打开它,它可以捕捉到在主线程序发生的磁盘IO读写、网络访问发生的意外,通常这个意外都是ANR(android not response),当时可以用这个工具检测出这些意外,代码就可以做调整:将磁盘读写、网络访问等耗时操作写到非主线程中,以提供更好的体验。但是在android7.0以后,这项功能被强制使用了。

在App之间共享文件

对于android7.0以后的系统,android框架强制使用StrictMode API 策略。这个策略禁止在app外暴露 “file://“URI。如果一个intent包含一个文件URI(以file://开头)离开你的app,那么这个app就报告FileUriExposedException异常。安装apk的功能实际上是系统提供的。Intent会带着我们的意图(包括apk的位置信息)离开我们的app,进入系统中,让系统中的应用来处理。这意味着apk文件需要在app间共享。
为了与其他应用共享文件,你应该发送"content://"URI ,并授予临时访问权限。授予这个临时访问权限的最签单方法就是使用FileProvider类。

注意:当然少不了android6.0之后要动态申请权限这一步。

使用FileProvider辅助完成安装任务

第一步:在manifest.xml申请以下权限,特别是第二个权限,如果缺少的话,在android8.0的手机上安装apk会失败

<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES" />

第二步:动态申请android.permission.READ_EXTERNAL_STORAGE权限。在android6.0之后,这个权限要动态申请,具体请看Github上的代码
第三步:在manifest.xml中声明FileProvider:

<?xml version="1.0" encoding="utf-8"?>
<manifest>
    <application>
    	......
        <provider
            android:name="android.support.v4.content.FileProvider"
            android:authorities="com.wong.myapp.UPDATE_APP_FILE_PROVIDER"
            android:exported="false"
            android:grantUriPermissions="true">
            <!-- 元数据 -->
            <meta-data
                android:name="android.support.FILE_PROVIDER_PATHS"
                android:resource="@xml/file_provider_paths" />
        </provider>
        ......
    </application>
</manifest>

android:name:provider可以直接使用v4包提供的FileProvider,或者自定义,但是自定义的provider一定要继承FileProvider,一般使用系统的就足够了。
android:authorities:这个值将会发布在content://下,即其他app通过content://com.wong.myapp.UPDATE_APP_FILE_PROVIDER来访问apk文件等信息,稍后将在代码中用到。
**android:exported:“false”**表示我们的provider不需要对外开放。
android:grantUriPermissions:“true”,此值为true,app才能获得临时共享权限。
元数据meta-data中:
**android:name=“android.support.FILE_PROVIDER_PATHS”**必须是这个名字。因为在
android.support.v4.content.FileProvider类里,要用这个key,获取我们的xml文件路径。

XmlResourceParser in = info.loadXmlMetaData(context.getPackageManager(), "android.support.FILE_PROVIDER_PATHS");

android:resource="@xml/file_provider_paths"指定配置可访问路径的xml的文件地址。
第四步:res/xml中定义对外暴露的文件夹路径
首先,在res资源目录入创建xml文件夹,然后在xml创建一份名为file_provider_paths.xml的文件,内容如下:

<?xml version="1.0" encoding="utf-8"?>
<resources xmlns:android="http://schemas.android.com/apk/res/android">
   <paths xmlns:android="http://schemas.android.com/apk/res/android">

       <!--代表外部存储区域的根目录下的文件 Environment.getExternalStorageDirectory()/DCIM/myapp目录-->
       <!--<external-path name="wong_DCIM" path="DCIM/myapp" />-->
       <!--代表外部存储区域的根目录下的文件 Environment.getExternalStorageDirectory()/Pictures/myapp目录-->
       <!--<external-path name="wong_Pictures" path="Pictures/myapp" />-->
       <!--代表app 外部存储区域根目录下的文件 Context.getExternalFilesDir(Environment.DIRECTORY_PICTURES)目录下的Pictures目录-->
       <!--/storage/emulated/0/Android/data/com.wong.myapp/files/Pictures-->
       <!--<external-files-path name="wong_external_files" path="Pictures" />-->
       <!--代表app 外部存储区域根目录下的文件 Context.getExternalCacheDir目录下的images目录-->
       <!--/storage/emulated/0/Android/data/com.wong.myapp/cache/images-->
       <!--<external-cache-path name="wong_external_cache" path="images" />-->
       <!--代表app 私有的存储区域 Context.getFilesDir()目录下的images目录 /data/user/0/com.wong.myapp/files/images-->
       <!--<files-path name="wong_private_files" path="images" />-->
       <!--代表app 私有的存储区域 Context.getCacheDir()目录下的images目录 /data/user/0/com.wong.myapp/cache/images-->
       <!--<cache-path name="wong_private_cache" path="images" />-->

       <root-path name="root_path" path="."/>

   </paths>

</resources>

注意:
在android8.0以上的手机上,读取共享文件时,如apk更新的编程中,我们把apk下载后,要安装时,去读取这个apk文件就出现如下错误:

Failed to find configured root that contains
...

所以一定要在file_provider_paths.xml文件中添加root-path 标签,即:

<root-path name="root_path" path="."/>

第五步:生成content://类型的Uri
使用以下代码只能生成格式为file://xxx的Uri,如果是android7.0之前的就用file://形式访问,获取Uri方式如下:

File picFile = xxx;
Uri picUri = Uri.fromFile(picFile);

android7.0以后的必须通过FileProvider.getUriForFile方法来生成content://xxx类型的Uri:

扫描二维码关注公众号,回复: 5538691 查看本文章

File apkFile = new File(Environment.getExternalStorageDirectory(), "update_folder/update.apk");
Uri fileUri = FileProvider.getUriForFile(MainActivity.this, "com.wong.myapp.UPDATE_APP_FILE_PROVIDER", apkFile);

getUriForFile:第一个参数是Context;第二个参数,就是我们之前在manifest#provider中定义的android:authorities属性的值;第三个参数是File

第六步:给Uri授予临时权限

intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION
               | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);

第七步:使用Intent传递Uri

intent.setDataAndType(fileUri, "application/vnd.android.package-archive");
startActivityForResult(intent, 100);

安装apk的核心代码:

 private void installApk() {

        Intent intent = new Intent(Intent.ACTION_VIEW);
        intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        intent.addCategory(Intent.CATEGORY_DEFAULT);
        //重新构造Uri:content://
        Uri fileUri;
        File apkFile = new File(Environment.getExternalStorageDirectory(), "update_folder/update.apk");
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            fileUri = FileProvider.getUriForFile(MainActivity.this, "com.wong.myapp.UPDATE_APP_FILE_PROVIDER", apkFile);
            //授予目录临时共享权限
            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
        } else {
            fileUri = Uri.fromFile(apkFile);
        }
        intent.setDataAndType(fileUri, "application/vnd.android.package-archive");
        //使用Intent传递Uri
        startActivity(intent);
    }

完整代码

猜你喜欢

转载自blog.csdn.net/weixin_40763897/article/details/88295946