Android:8.0中未知来源安装权限

1.问题的现象

在我们测试APK升级的时候,会遇到8.0在下载完成 以后就没有然后了。没有弹起安装页面,不执行安装逻辑。但是在8.0之前的版本,可以正常下载可以正常弹起安装页面。

2.问题的分析

通过查询资料得到,Android8.0以后增加一个未知来源权限。

  • 将设置---安全中的允许安装未知来源应用取消了(由于国内的手机系统的高度定制,该选择项的位置有差异)

  • 在安装APK文件时新增 ,未知来源安装权限android.permission.REQUEST_INSTALL_PACKAGES
    也就是说,在安卓8.0以后(Android o)之前,设置中的允许安装来源是针对所有App的,只要开启了,那么所有位置来源的App都会安装,但是在8.0之后,将这个权限挪到每一个App的内部,这样大大的提高了手机的安全性,降低了流氓软件的安装概率。
    参考资料: Making it safer to get apps on Android O

3.解决办法

(1)、步骤一

在AndroidMainfest.xml清单文件中增加如下权限

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

(2)、步骤二

  • 我们通过ACTION_MANAGE_UNKNOWN_APP_SOURCES
    这个Action可以跳转到未知来源安装设置界面,引导用户去开启这个选项。
  • 我们可以通过PackageManager中canRequestPackageInstalls()来检测是否已经开启了未知来源安装权限。true表示获取了权限,false表示没有获取权限。为false时,安装过程会被中断,无法跳转到安装页面。
    所以,我们在下载完APK之后,可以按照下面的流程来处理
    流程图

以我项目中的为例子下载的代码

    //下载的代码
    private void downFile(final String versionUrl) {
      //创建一个下载提醒框
        ProgressDialog progressDialog = new ProgressDialog(this);
        progressDialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
        progressDialog.setCancelable(false);
        progressDialog.setTitle("下载中。。。。");
        progressDialog.setProgress(0);
        progressDialog.show();
        new Thread(new Runnable() {
            @Override
            public void run() {
                OkHttpClient build = new OkHttpClient.Builder().build();
                Request request = new Request.Builder().url(versionUrl).build();
                build.newCall(request).enqueue(new Callback() {
                    @Override
                    public void onFailure(Call call, IOException e) {

                    }

                    @Override
                    public void onResponse(Call call, Response response) throws IOException {
                        int length = (int) response.body().contentLength();
                        InputStream inputStream = response.body().byteStream();
                        progressDialog.setMax(length);
                        FileOutputStream fos = null;
                        if (inputStream != null) {
                            File file = new File(Environment.getExternalStorageDirectory().getAbsolutePath(), "jiuxing.apk");
                            fos = new FileOutputStream(file);
                            byte[] buf = new byte[1024];
                            int ch;
                            int process = 0;
                            while ((ch = inputStream.read(buf)) != -1) {
                                fos.write(buf, 0, ch);
                                process += ch;
                                progressDialog.setProgress(process); // 实时更新进度了
                            }
                            if (fos != null) {
                                fos.flush();
                                fos.close();
                            }
                            //关闭dialog防止泄露
                            progressDialog.dismiss();
                            // 下载完成后安装 代码省略
                            ...............

                        }
                    }
                });
            }
        }).start();
    }
  • 下面的逻辑可以在我们的主页中实现 可以直接使用startActivityForResult并在onActivityResult中解析数据
/**
     * 打开安装包
     */
    private void openAPKFile() {
        String mimeDefault = "application/vnd.android.package-archive";

        File apkFile = null;
        if (!TextUtils.isEmpty(mApkUri)) {
            //mApkUri是apk下载完成后在本地的存储路径
            apkFile = new File(Uri.parse(mApkUri).getPath());
        }
        if (apkFile == null) {
            return;
        }

        try {
            Intent intent = new Intent(Intent.ACTION_VIEW);
            intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
            //兼容7.0
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
                //这里牵涉到7.0系统中URI读取的变更
                Uri contentUri = FileProvider.getUriForFile(mActivity, getPackageName() + ".fileprovider", apkFile);
                intent.setDataAndType(contentUri, mimeDefault);
                //兼容8.0
                if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
                    boolean hasInstallPermission = getPackageManager().canRequestPackageInstalls();
                    if (!hasInstallPermission) {
                        startInstallPermissionSettingActivity();
                        return;
                    }
                }
            } else {
                intent.setDataAndType(Uri.fromFile(apkFile), mimeDefault);
            }
            if (getPackageManager().queryIntentActivities(intent, 0).size() > 0) {
                //如果APK安装界面存在,携带请求码跳转。使用forResult是为了处理用户 取消 安装的事件。外面这层判断理论上来说可以不要,但是由于国内的定制,这个加上还是比较保险的
                startActivityForResult(intent, 2);
            }
        } catch (Throwable e) {
            e.printStackTrace();
        }
    }

    /**
     * 跳转到设置-允许安装未知来源-页面
     */
    @RequiresApi(api = Build.VERSION_CODES.O)
    private void startInstallPermissionSettingActivity() {
        //后面跟上包名,可以直接跳转到对应APP的未知来源权限设置界面。使用startActivityForResult 是为了在关闭设置界面之后,获取用户的操作结果,然后根据结果做其他处理
        Intent intent = new Intent(Settings.ACTION_MANAGE_UNKNOWN_APP_SOURCES, Uri.parse("package:" + getPackageName()));
        startActivityForResult(intent, 1);
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if (resultCode == RESULT_OK) {
            if (requestCode == 1) {
                openAPKFile();
            }
        } else {
            if (requestCode == 1) {
                //下午4:31 8.0手机位置来源安装权限
                if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
                    boolean hasInstallPermission = getPackageManager().canRequestPackageInstalls();
                    if (!hasInstallPermission) {
                        LogUtils.e(TAG, "没有赋予 未知来源安装权限");
                        showUnKnowResourceDialog();
                    }
                }
            } else if (requestCode == 2) {
                // CnPeng 2018/8/2 下午4:31 在安装页面中退出安装了
                LogUtils.e(TAG, "从安装页面回到欢迎页面--拒绝安装");

                showApkInstallDialog();
            }
        }
    }

    /**
 
     * 功用:弹窗请安装APP的弹窗
     * 说明:8.0手机升级APK时获取了未知来源权限,并跳转到APK界面后,用户可能会选择取消安装,所以,再给一个弹窗
     */
    private void showApkInstallDialog() {
        final CustomAlertDialog installDialog = new CustomAlertDialog(mActivity);
        installDialog.setCancelable(false);
        DialogInstallApkBinding binding = DataBindingUtil.inflate(getLayoutInflater(), R.layout.dialog_install_apk, null, false);
        installDialog.setView(binding.getRoot());
        installDialog.show();

        binding.ivIKnowBt2.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //再次回到安装界面
                openAPKFile();
            }
        });

        binding.tvInstallNext.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                installDialog.dismiss();

                //CnPeng 2018/8/2 下午5:28  使用自定义方法关闭全部activity
                ActivitiesCollector.finishAll();
            }
        });
    }

    /**
     * 说明:8.0系统中升级APK时,如果跳转到了 未知来源权限设置界面,并且用户没用允许该权限,会弹出此窗口
    **/
    private void showUnKnowResourceDialog() {
        final CustomAlertDialog alertDialog = new CustomAlertDialog(mActivity);
        alertDialog.setCancelable(false);

        DialogUnknowResourceBinding binding = DataBindingUtil.inflate(getLayoutInflater(), R.layout.dialog_unknow_resource, null, false);
        alertDialog.setView(binding.getRoot());
        alertDialog.show();

        binding.ivIKnowBt.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //兼容8.0
                if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
                    boolean hasInstallPermission = getPackageManager().canRequestPackageInstalls();
                    if (!hasInstallPermission) {
                        startInstallPermissionSettingActivity();
                    }
                }

                alertDialog.dismiss();
            }
        });
    }

4.个人总结

在关注新版本特性时,不能只关注新控件,其他系统级的变更必须高度重视。这次的8.0安装权限变更就是一个教训啊!

参考资料

Making it safer to get apps on Android O

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,036评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,046评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,411评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,622评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,661评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,521评论 1 304
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,288评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,200评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,644评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,837评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,953评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,673评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,281评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,889评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,011评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,119评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,901评论 2 355

推荐阅读更多精彩内容

  • 一、问题现象 在测试APK升级逻辑时,偶然发现在8.0系统的手机中,APK下载完就没有然后了,没有弹出安装界面,不...
    CnPeng阅读 22,340评论 18 28
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,133评论 25 707
  • 自己想要过什么样的生活,全凭自己的选择。人生处处是选择,选择每天吃什么,是否运动,都是自己说了算。 选择权在我们的...
    紫翼天葵阅读 166评论 1 4
  • 「1」 正月十五,上元佳节。 而黎沧国的朝堂之上却是一片凝重。金座上人一脸肃杀之气,将手里的奏折狠狠摔在了地上。帝...
    终朝采绿zql阅读 649评论 2 2
  • 14-4-21 每个孩子都有着不同于他人的性格,他们在学前期的成长道路上,会有许多值得老师与父母正确引导的地方。 ...
    耀仙阅读 870评论 1 6