Android 应用内下载Apk, 安装,适配8.0

\color{#008484}{ 下载安装需要配置的权限}

<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
<!--未知来源安装权限 -->
<uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES" />

\color{#008484}{ 一、请求存储权限}

   /**
    * 我这里用了别人的请求权限框架 AndPemission
    * Github地址:https://github.com/yanzhenjie/AndPermission
    */
    if (AndPermission.hasPermissions(this, Permission.Group.STORAGE)) {
        // 有存储权限, 开启服务下载
        startService();  
    } else {
       // 无存储权限, 则请求权限
       AndPermission.with(this).runtime().permission(Permission.Group.STORAGE).onGranted(data -> {
            if (CollectionUtils.isEmpty(data)) {
               return;
            }
            // 用户允许了权限, 开启服务下载
            startService();      
       }).start();
    }

\color{#008484}{二、启动服务开始下载Apk,注意适配安卓8.0}

   Intent intent = new Intent(this, ApkDownloadService.class);
   // 给Service传值, 我这里直接把整个Bean传过去了, 其实只需要个下载地址和当前下载的版本号
   intent.putExtra(IntentKey.BEAN, newVersionBean);
   if (Build.VERSION.SDK_INT >= 26) {
       // Android8.0适配
       startForegroundService(intent);
   } else {
       startService(intent);
   }

\color{#008484}{在Service的onStartCommand方法中处理下载}

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        // android8.0适配: 被启动的Service创建服务后的五秒内要调用startForground(0, new Notification())
        // 如果不调用或调用时间超过5秒会抛出一个ANR
        // 调用startForground用到了通知,android8.0通知又必须要设置通知渠道
        // 创建通知渠道并运行服务到前台
        createNotificationChannel();
        // 获取Intent传值信息
        mVersionInfo = (QueryVersionsVo) intent.getSerializableExtra(IntentKey.BEAN);
        // 开始异步任务下载Apk
        downloadApk();
        return super.onStartCommand(intent, flags, START_STICKY);
    }

\color{#008484}{Service的几个方法}

    private void createNotificationChannel() {
        // 这里的id输入自己的项目的包名
        String ID = "com.***.***";
        String NAME = "Channel One";
        Intent intent = new Intent(this, HomeActivity.class);
        PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, 0);
        // 创建服务对象
        NotificationCompat.Builder notification; 
        NotificationManager manager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
        // Android8.0要求必须创建通知渠道
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            NotificationChannel channel = new NotificationChannel(ID, NAME, NotificationManager.IMPORTANCE_HIGH);
            channel.enableLights(false);
            channel.setShowBadge(false);
            channel.setLockscreenVisibility(Notification.VISIBILITY_PUBLIC);
            manager.createNotificationChannel(channel);
        }
        notification = new NotificationCompat.Builder(this, ID);
        notification.setContentTitle("更新提示")
                .setContentText("正在下载最新版本..")
                .setWhen(System.currentTimeMillis())
                .setSmallIcon(R.mipmap.ic_launcher_esp)
                .setLargeIcon(BitmapFactory.decodeResource(getResources(), R.mipmap.ic_launcher_esp))
                .setContentIntent(pendingIntent)
                .build();
        Notification notification1 = notification.build();
        startForeground(1, notification1);
    }


    public void downloadApk() {
        // 设置最新安装包名称
        StringBuilder builder = new StringBuilder("your_app_name-");
        if (!TextUtils.isEmpty(mVersionInfo.getPushVersions())) {
            builder.append(mVersionInfo.getPushVersions());
        } else {
            builder.append(System.currentTimeMillis());
        }
        builder.append(".apk");

        // 设置apk所在目录
        File baseFile = ContextHolder.getContext().getExternalFilesDir("yc_team");
        if (!baseFile.exists()) {
            baseFile.mkdirs();
        }
        mApkFile = new File(baseFile.getPath(), builder.toString());
        // 最终apk目录  文件管理-手机存储-Android-data-应用包名-yc_team-***.apk
        if (mApkFile.exists()) {
            mApkFile.delete();
        }

        // 开始异步下载
        mAsyncTask = new DownApkAsyncTask();
        mAsyncTask.execute();
    }

    @SuppressLint("StaticFieldLeak")
    private class DownApkAsyncTask extends AsyncTask<Void, Long, Void> {

        @Override
        protected Void doInBackground(Void... voids) {
            
            HttpURLConnection httpConnection = null;
            InputStream is = null;
            FileOutputStream fos = null;
            int updateTotalSize;
            URL url;
            try {
                url = new URL(mVersionInfo.getUrl());
                httpConnection = (HttpURLConnection) url.openConnection();
                httpConnection.setConnectTimeout(60000);
                httpConnection.setReadTimeout(60000);
                if (httpConnection.getResponseCode() != 200) {
                    return null;
                }
                updateTotalSize = httpConnection.getContentLength();

//                mApkFile.createNewFile();
                is = httpConnection.getInputStream();
                fos = new FileOutputStream(mApkFile, false);
                byte[] buffer = new byte[4096];

                int readSize;
                int currentSize = 0;

                while ((readSize = is.read(buffer)) > 0) {
                    fos.write(buffer, 0, readSize);
                    currentSize += readSize;

                    int finalCurrentSize = currentSize;
                    int finalUpdateTotalSize = updateTotalSize;
                    // 这里可以发消息实时通知进度
                    handler.post(() -> {
                        // 用finalCurrentSize和finalUpdateTotalSize计算出进度
                        
                    });
                }
                // 下载完成, 通知安装
                installApk();
            } catch (Exception e) {
                e.printStackTrace();
                return null;
            } finally {
                if (httpConnection != null) {
                    httpConnection.disconnect();
                }
                if (is != null) {
                    try {
                        is.close();
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
                if (fos != null) {
                    try {
                        fos.close();
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            }
            return null;
        }
    }

    /**
     * 我这里是发了个EventBus,通知Activity可以安装Apk了
     */
    private void installApk() {
        handler.post(() -> {
            EventBusMode mode = new EventBusMode(EventBusType.ESP_DOWNLOAD_APK_SUCCESS);
            mode.setTempStr(mApkFile.getAbsolutePath());
            EventBus.getDefault().post(mode);
            // 下载完成,关闭当前服务
            stopSelf();
        });
    }

    /**
     * 在Service结束时, 停止异步下载
     */
    @Override
    public void onDestroy() {
        if (mAsyncTask != null) {
            mAsyncTask.cancel(true);
        }
        super.onDestroy();
    }

\color{#008484}{Activity收到通知,开始请求未知来源安装权限}

   if (Build.VERSION.SDK_INT < Build.VERSION_CODES.O) {
        // Android8.0之前,直接安装Apk
        installApk();
        return;
   }
   boolean haveInstallPermission = getPackageManager().canRequestPackageInstalls();
   if (!haveInstallPermission) {
        // 权限没有打开则提示用户去手动打开
        Uri packageURI = Uri.parse("package:" + getPackageName());
        Intent intent = new Intent(Settings.ACTION_MANAGE_UNKNOWN_APP_SOURCES, packageURI);
        startActivityForResult(intent, 1001);
   }
    /**
     * 未知来源安装权限申请回调
     */
    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if (resultCode != RESULT_OK) {
            return;
        }
        if (requestCode == 1001 && Build.VERSION.SDK_INT >=   Build.VERSION_CODES.O) {
            // 未知来源安装应用权限开启
            boolean haveInstallPermission = getPackageManager().canRequestPackageInstalls();
            if (haveInstallPermission) {
                installApk();
            }
        }
    }

    /**
     * 安装最新Apk
     */
    private void installApk() {
        // Service发的通知中的文件绝对路径
        File file = new File(mNewApkFilePath);
        try {
            // 这里有文件流的读写,需要处理一下异常
            Intent intent = new Intent(Intent.ACTION_VIEW);
            intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                //如果SDK版本>=24,即:Build.VERSION.SDK_INT >= 24
                String packageName = context.getApplicationContext().getPackageName();
                String authority = new StringBuilder(packageName).append(".provider").toString();
                Uri uri = FileProvider.getUriForFile(context, authority, file);
                intent.setDataAndType(uri, "application/vnd.android.package-archive");
            } else {
                Uri uri = Uri.fromFile(file);
                intent.setDataAndType(uri, "application/vnd.android.package-archive");
            }
            context.startActivity(intent);
        } catch (Exception e) {
            e.printStackTrace();
            // 安装的时候会进行版本自动检测,更新版本小于已有版本,是会走当前异常的,注意!
        }
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,214评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,307评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,543评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,221评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,224评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,007评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,313评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,956评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,441评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,925评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,018评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,685评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,234评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,240评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,464评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,467评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,762评论 2 345

推荐阅读更多精彩内容