之前有写过利用腾讯Bugly实现APP的热更新以及版本升级Android 热更新框架Bugly-9步完成热更新/自动更新/异常上报分析,今天来讲一下不借助第三方的应用升级。

演示效果:

android app 升级原理 android升级软件_应用升级

android app 升级原理 android升级软件_android_02

原理:

1.将新版本上传到自己的服务器,有服务器将最新版本信息记录

2.当用户打开app或者手动触发版本检查时向服务器请求版本信息以及最新版本apk的下载地址

3.判断当前版本是不是最新版本,如果不是则通过下载地址下载apk

4.下载完成后吊起安装程序进行安装覆盖

5.实现了版本更新

本次例子用到框架:

1.easypermissions  权限控制

2.gson  json对象解析

3.xUtils-2.6.14.jar 网络请求

所用权限(注意最后一个权限):

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

    <!--以下这个权限如果不加,在高版本的android手机上如果没有开启运行未知来源的安装,将会在下载完成后无法调起安装程序-->
    <uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES" />

注意点:

 1."android.permission.REQUEST_INSTALL_PACKAGES"这个权限必须加上,否则可能会在apk下载完成后不能吊起安装程序(一闪而逝)

2.Android8以上如果要在通知栏显示下载进度,需要进行notification的适配Android8.0 notification channel

3.在Android6.0以上需要进行权限的申请

4.Android7.0以上文件读取需要通过FileProvider进行操作关于 Android 7.0 适配中 FileProvider 部分的总结

下面看主要代码:

MainActivity

package cn.humanetplan.updateappdemo;

import android.Manifest;
import android.content.Intent;
import android.support.annotation.NonNull;
import android.view.View;
import android.widget.TextView;

import com.google.gson.Gson;
import com.lidroid.xutils.HttpUtils;
import com.lidroid.xutils.exception.HttpException;
import com.lidroid.xutils.http.RequestParams;
import com.lidroid.xutils.http.ResponseInfo;
import com.lidroid.xutils.http.callback.RequestCallBack;
import com.lidroid.xutils.http.client.HttpRequest;

import java.util.List;

import pub.devrel.easypermissions.EasyPermissions;

;

public class MainActivity extends BaseActivity implements EasyPermissions.PermissionCallbacks{
    TextView tv_check, tv_versionName;

    @Override
    public void DoSthBeforeInflate() {

    }

    @Override
    protected int setContentLayout() {
        return R.layout.activity_main;
    }

    @Override
    protected void init() {
        tv_check = findViewById(R.id.tv_check);
        tv_versionName = findViewById(R.id.tv_versionName);
        tv_versionName.setText("当前版本V"+BuildConfig.VERSION_NAME);
        tv_check.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                CheckVersion();
            }
        });
    }
    VersionBean versionBean;
    private void CheckVersion() {
        tipDialog.show();
        HttpUtils httpUtils = new HttpUtils();
        RequestParams params = new RequestParams();
        httpUtils.send(HttpRequest.HttpMethod.GET, "http://47.107.173.197/Android/GetVersionInfo.php", new RequestCallBack<String>() {
            @Override
            public void onSuccess(ResponseInfo<String> responseInfo) {
                tipDialog.dismiss();
                try {
                    Gson gson = new Gson();

                    versionBean = gson.fromJson(responseInfo.result, VersionBean.class);
                    if (versionBean != null && versionBean.isSuccess()) {
                        DialogUtils.ShowTipsDialog(MainActivity.this, "发现新版本,是否立即更新?", "当前版本V"+versionBean.getVersionName()+"\n"+versionBean.getRemark(), new DialogReturnListner() {
                            @Override
                            public void onResultReturn(String... params) {

                            }

                            @Override
                            public void onResultReturn(int p1, String... params) {

                            }

                            @Override
                            public void onResultReturn(boolean p1, String... params) {
                                if (p1) {
                                    if (EasyPermissions.hasPermissions(MainActivity.this, Manifest.permission.READ_EXTERNAL_STORAGE,Manifest.permission.WRITE_EXTERNAL_STORAGE)){
                                        StartUpdate(versionBean);
                                    }else {
                                        EasyPermissions.requestPermissions(MainActivity.this,"升级程序将App下载到手的过程中需要用到手机文件操作权限,请同意后才能进行正常升级",1234,Manifest.permission.READ_EXTERNAL_STORAGE,Manifest.permission.WRITE_EXTERNAL_STORAGE);
                                    }


                                }
                            }
                        });
                    }
                } catch (Exception e) {

                }

            }

            @Override
            public void onFailure(HttpException e, String s) {
                tipDialog.dismiss();
            }
        });

    }
    private void StartUpdate(VersionBean versionBean) {
        if (versionBean==null){
            return;
        }
        ToastUtils.showToast(MainActivity.this, "正在更新...");
        Intent intent=new Intent(MainActivity.this,ServiceLoadNewVersion.class);
        intent.putExtra("path",versionBean.getApkPath());
        startService(intent);
    }

    @Override
    public void onPermissionsGranted(int requestCode, @NonNull List<String> perms) {
        if (requestCode==1234){
            if (perms.contains(Manifest.permission.READ_EXTERNAL_STORAGE) && perms.contains(Manifest.permission.WRITE_EXTERNAL_STORAGE)){
                StartUpdate(versionBean);
            }
        }
    }

    @Override
    public void onPermissionsDenied(int requestCode, @NonNull List<String> perms) {
        ToastUtils.showToast(this,"没有获取相关的权限,无法正常操作!");
    }

    @Override
    public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);

        // Forward results to EasyPermissions
        EasyPermissions.onRequestPermissionsResult(requestCode, permissions, grantResults, this);
    }
}

 ServiceLoadNewVersion

package cn.humanetplan.updateappdemo;

import android.app.Notification;
import android.app.NotificationChannel;
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.app.Service;
import android.content.Intent;
import android.graphics.BitmapFactory;
import android.net.Uri;
import android.os.Build;
import android.os.Environment;
import android.os.Handler;
import android.os.IBinder;
import android.os.Message;
import android.support.v4.app.NotificationCompat;
import android.support.v4.content.FileProvider;
import android.util.Log;
import android.widget.RemoteViews;

import com.lidroid.xutils.HttpUtils;
import com.lidroid.xutils.exception.HttpException;
import com.lidroid.xutils.http.HttpHandler;
import com.lidroid.xutils.http.RequestParams;
import com.lidroid.xutils.http.ResponseInfo;
import com.lidroid.xutils.http.callback.RequestCallBack;

import java.io.File;

/**
 * 下载 新 版本 的 服务
 */
public class ServiceLoadNewVersion extends Service {


    private RemoteViews remoteViews = null;
    private Notification notification = null;
    private NotificationManager notificationManager = null;
    private PendingIntent pReDownLoadIntent = null;

    private Handler myHandler;

    // notification id
    private final int NOTIFICATION_ID = 1000;
    private final int START = 1001;
    private final int LOADING = 1002;
    private final int FINISHED = 1003;
    private final int LOAD_ERROR = 1004;
    private String url;
    private String filePath;

    String apkName = "updateTest.apk";
    String loagPath = "";

    public ServiceLoadNewVersion() {
    }

    @Override
    public IBinder onBind(Intent intent) {
        // TODO: Return the communication channel to the service.
        throw new UnsupportedOperationException("Not yet implemented");
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        try {
            String path = intent.getStringExtra("path");
            url = path;
            myHandler = new MyHandler();
            initUrls();

            initNotification();

            // 开始 下载
            new DownLoadThread(url, filePath).start();
        } catch (Exception e) {

        }
        return super.onStartCommand(intent, flags, startId);
    }

    /**
     * 开始 下载
     */
    private void initUrls() {
        File file = new File(
                Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS)
                , apkName);
        if (file.exists()) {
            file.delete();
        }
        filePath = file.getAbsolutePath();
    }

    NotificationCompat.Builder builder = null;

    /**
     * 配置 通知栏显示 样式
     */
    private void initNotification() {
        String id = "chanel_update";
        String name = "水务集团";
        notificationManager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            //如果是8以上的系统。需要传一个channelId.
            builder = new NotificationCompat.Builder(this, id);
        } else {
            builder = new NotificationCompat.Builder(this);
        }

        builder.setContentTitle(getResources().getString(R.string.app_name) + "新版本下载").
                setContentText("下载进行中...")
                .setVibrate(new long[]{0})
                .setLargeIcon(BitmapFactory.decodeResource(getResources(), R.mipmap.applogo))
                .setSmallIcon(R.mipmap.applogo)
                .setProgress(100, 0, false);


        // 下载 失败 重新 下载 的 PendingIntent
        Intent reDownLoadIntent = new Intent(this, this.getClass());
        reDownLoadIntent.putExtra("path", loagPath);
        pReDownLoadIntent = PendingIntent.getService(this, 200, reDownLoadIntent, PendingIntent.FLAG_CANCEL_CURRENT);


        remoteViews = new RemoteViews(getPackageName(), R.layout.view_download_notification);
        remoteViews.setTextViewText(R.id.textViewTitle, "正在下载");
        remoteViews.setTextViewText(R.id.textViewProgress, "进度0%");
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {//android8.0以上通知的适配
            NotificationChannel mChannel = new NotificationChannel(id, name, NotificationManager.IMPORTANCE_LOW);
            mChannel.enableVibration(false);
            mChannel.setVibrationPattern(new long[]{0});

            notificationManager.createNotificationChannel(mChannel);
            notification = builder.build();
        } else {

            notification = builder.build();
        }
    }


    class MyHandler extends Handler {
        @Override
        public void handleMessage(Message msg) {
            switch (msg.what) {
                case START:
                    notificationManager.notify(NOTIFICATION_ID, notification);

                    break;
                case LOADING:
                    builder.setProgress(100, msg.arg1, false);
                    builder.setContentText("下载进行中" + msg.arg1 + "/100");
                    notification = builder.build();
                    notificationManager.notify(NOTIFICATION_ID, notification);
                    //关键部分,如果你不重新更新通知,进度条是不会更新的
                    break;
                case FINISHED:

                    notification.flags |= Notification.FLAG_AUTO_CANCEL;
//                    //关键部分,如果你不重新更新通知,进度条是不会更新的
                    notificationManager.notify(NOTIFICATION_ID, notification);
                    notificationManager.cancel(NOTIFICATION_ID);
                    installApkNew(null);

                    break;
                case LOAD_ERROR:
                    builder.setContentTitle("新版本下载失败");
                    builder.setContentText("下载失败,点击重新下载!");
                    notification.contentIntent = pReDownLoadIntent;
//                    notification.flags = Notification.FLAG_NO_CLEAR; // 点击通知 不消失
                    //关键部分,如果你不重新更新通知,进度条是不会更新的
                    notificationManager.notify(NOTIFICATION_ID, notification);
                    break;
            }
        }
    }

    //安装apk

    protected void installApkNew(Uri uri) {

        try {
            File file = new File(filePath);
            Intent intent = new Intent(Intent.ACTION_VIEW);
            // 由于没有在Activity环境下启动Activity,设置下面的标签
            intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
            if (Build.VERSION.SDK_INT >= 24) { //判读版本是否在7.0以上
                //参数1 上下文, 参数2 Provider主机地址 和配置文件中保持一致   参数3  共享的文件
                Uri apkUri =
                        FileProvider.
                                getUriForFile(getApplicationContext(),
                                        BuildConfig.APPLICATION_ID + ".provider",
                                        file);

                //添加这一句表示对目标应用临时授权该Uri所代表的文件
                intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
                intent.setDataAndType(apkUri, "application/vnd.android.package-archive");
            } else {
                Uri mUri = Uri.fromFile(file);
                mUri = Uri.parse(mUri.toString().replace("content", "file"));
                intent.setDataAndType(mUri,
                        "application/vnd.android.package-archive");
            }


            startActivity(intent);
        } catch (Exception e) {

            e.printStackTrace();
            Log.i("ServiceLoadNewVersion", "exc  " + e.toString());
        }
    }


    class DownLoadThread extends Thread {
        private String url;
        private String filePath;

        public DownLoadThread(String url, String filePath) {
            this.url = url;
            this.filePath = filePath;
        }

        @Override
        public void run() {

            HttpUtils http = new HttpUtils();
            RequestParams params = new RequestParams();
            HttpHandler handler = http.download(
                    url,//url
                    filePath, // 文件保存路径,
                    params,
                    true, // 如果目标文件存在,接着未完成的部分继续下载。服务器不支持RANGE时将从新下载。
                    false, // 如果从请求返回信息中获取到文件名,下载完成后自动重命名。
                    new RequestCallBack<File>() {
                        @Override
                        public void onStart() {
                            myHandler.sendEmptyMessage(START);
                        }

                        @Override
                        public void onLoading(long total, long current, boolean isUploading) {
                            Message message = myHandler.obtainMessage();
                            message.what = LOADING;
                            message.arg1 = (int) ((float) current / (float) total * 100);
                            myHandler.sendMessage(message);


                        }

                        @Override
                        public void onSuccess(ResponseInfo<File> responseInfo) {
                            myHandler.sendEmptyMessage(FINISHED);
                        }


                        @Override
                        public void onFailure(HttpException error, String msg) {
                            myHandler.sendEmptyMessage(LOAD_ERROR);
                        }
                    });
        }
    }
}

Manifest

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="cn.humanetplan.updateappdemo">
    <uses-permission android:name="android.permission.INTERNET"/>
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>

    <!--以下这个权限如果不加,在高版本的android手机上如果没有开启运行未知来源的安装,将会在下载完成后无法调起安装程序-->
    <uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES" />
    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/Theme.AppCompat.Light.NoActionBar">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

        <!--android 7.0以上文件访问需要通过FileProvider进行,否则会报错-->
        <provider
            android:name="android.support.v4.content.FileProvider"
            android:authorities="${applicationId}.provider"
            android:exported="false"
            android:grantUriPermissions="true">
            <meta-data
                android:name="android.support.FILE_PROVIDER_PATHS"
                android:resource="@xml/filepaths"/>
        </provider>
        <service android:name=".ServiceLoadNewVersion"
            android:enabled="true"
            android:exported="true"/>
    </application>

</manifest>

代码比较简单,没有太多注释。

完整代码: UpdateAppDemo.zip

Github:https://github.com/shouPol/UpdateDemo