Android程序中通过PackageInstaller实现版本更新

/ 5

Android更新app就不需要像iOS那样依赖于AppStore了,完全可以将 apk 安装包放在自己的服务器,每次启动app时去检查服务器的版本信息。如果高于本地版本就下载安装新版本。

版本控制

versionCode:类似iOS里的Build构建号,不对外显示,只是内部使用和应用商店更新时去修改构建号,一般每次更新版本就递增1。

versionName:类似iOS里的Version版本号,这个才是展示在应用商店和app界面上的版本号,一般我们更新版本时比较这个就行了。

请求更新版本接口后,我们可以获取到类似的返回信息,然后和本地应用版本进行比较。

php脚本:

<?php
exit(json_encode([
    "version" => "1.0.1",
    "apk_url" => "https://blog.6ag.cn/study/Mobilesafe.apk",
    "des" => "新版本上线了,快来下载吧!!!"
];));

返回的json:

{
    "code": "1.0.1",
    "apkurl": "https://blog.6ag.cn/study/Mobilesafe.apk",
    "des": "新版本上线了,快来下载吧!!!"
}

获取本地版本号

/**
 * 获取当前应用的版本号
 *
 * @return 版本号
 */
private String getVersionName() {
    PackageManager pm = getPackageManager();
    // 第一个参数:应用程序的包名
    // 第二个参数:指定信息的标签,0表示获取基础信息,比如包名、版本号。要想获取权限等信息必须要通过标签指定。
    try {
        PackageInfo info = pm.getPackageInfo(getPackageName(), 0);
        return info.versionName;
    } catch (PackageManager.NameNotFoundException e) {
        e.printStackTrace();
    }
    return "";
}

安装APK

如果有则开启子线程去下载apk安装文件,并获取到apk文件下载后的保存路径,然后启动Android系统自带的应用 PackageInstaller 去进行安装,通过分析这个应用程序的 AndroidManifest ,我们可以使用隐式 Intent 来启动 PackageInstaller 安装apk的 activity 。

/**
 * 安装下载的新版本apk
 * @param apkPath apk存放路径
 */
private void installAPK(String apkPath) {
    Intent intent = new Intent();
    intent.setAction("android.intent.action.VIEW");
    intent.addCategory("android.intent.category.DEFAULT");
    intent.setDataAndType(Uri.fromFile(new File(apkPath)), "application/vnd.android.package-archive");
    startActivityForResult(intent, MY_INSTALL_REQUEST);
}

注意我们需要使用带响应结果的启动方法:startActivityForResult(..),因为用户有可能会取消安装。

/**
 * 跳转activity后的回调结果
 */
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    switch (requestCode) {
        case MY_INSTALL_REQUEST:
            if (resultCode == RESULT_CANCELED) {
                // 应用安装被取消,跳转到主界面
            }
            break;
    }
}

安装完毕后会自动退出应用程序,如果取消安装则进入主界面或继续使用当前版本,这样就完成了整个升级过程。

完整代码:

PHP部分:

<?php
$result = [
    "err_msg" => "success",
    "info" => "获取版本信息成功",
    "data" => [
        "version" => "1.0.1",
        "description" => "1.修复已知bug\n2.兼容Android4.2系统",
        "url" => "http://download.baokan.tv/baokan.apk"
    ]
];
exit(json_encode($result));

Java部分:

private static final String TAG = "MainActivity";
private static final int PERMISSIONS_REQUEST_WRITE_EXTERNAL_STORAGE = 10000; // 写入SD卡权限
private ProgressDialog mDownloadDialog;    // 加载进度
private String serverVersion; // 服务器版本号
private String description;   // 新版本更新描述
private String apkUrl;        // 新版本apk下载地址

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    // 检查版本更新
    checkVersion();

}

/**
 * 检查是否有新版本
 */
private void checkVersion() {

    NetworkUtils.shared.get(APIs.UPDATE, new HashMap<String, String>(), new NetworkUtils.StringCallback() {
        @Override
        public void onError(Call call, Exception e, int id) {
            ProgressHUD.showInfo(mContext, "您的网络不给力");
        }

        @Override
        public void onResponse(String response, int id) {
            try {
                JSONObject jsonObject = new JSONObject(response);
                if (jsonObject.getString("err_msg").equals("success")) {
                    JSONObject versionInfo = jsonObject.getJSONObject("data");
                    serverVersion = versionInfo.getString("version");
                    description = versionInfo.getString("description");
                    apkUrl = versionInfo.getString("url");

                    // 更新版本
                    showUpdateDialog();
                }
            } catch (JSONException e) {
                e.printStackTrace();
                ProgressHUD.showInfo(mContext, "数据解析异常");
            }
        }
    });

}

/**
 * 弹出对话框更新app版本
 */
protected void showUpdateDialog() {

    // 检查是否是新版本
    String currentVersion = BaoKanApp.app.getVersionName();
    if (currentVersion.equals(serverVersion)) {
        return;
    }

    AlertDialog.Builder builder = new AlertDialog.Builder(this);
    builder.setTitle("发现新版本:" + serverVersion);
    builder.setIcon(R.mipmap.ic_launcher);
    builder.setMessage(description);
    builder.setCancelable(false);
    builder.setPositiveButton("立即更新", new DialogInterface.OnClickListener() {

        @Override
        public void onClick(DialogInterface dialog, int which) {
            // 判断是否有写入SD权限
            if (ContextCompat.checkSelfPermission(mContext,
                    Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
                // 申请权限
                ActivityCompat.requestPermissions(mContext,
                        new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},
                        PERMISSIONS_REQUEST_WRITE_EXTERNAL_STORAGE);
            } else {
                // 有写入权限直接下载apk
                downloadAPK();
            }
        }
    });
    builder.setNegativeButton("取消", null);
    builder.show();
}

/**
 * 运行时权限请求回调结果
 */
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
    switch (requestCode) {
        case PERMISSIONS_REQUEST_WRITE_EXTERNAL_STORAGE:
            if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                downloadAPK();
            } else {
                Toast.makeText(getApplicationContext(), "你没有文件写入权限", Toast.LENGTH_SHORT).show();
            }
            break;
    }
}

/**
 * 下载新版本
 */
protected void downloadAPK() {

    // apk文件保存路径
    String apkPath = null;
    String apkName = "baokan" + serverVersion + ".apk";
    if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {
        apkPath = Environment.getExternalStorageDirectory().getAbsolutePath();
    }

    if (apkPath == null) {
        ProgressHUD.showInfo(mContext, "您的手机没有SD卡");
        return;
    }

    // 弹出下载进度会话框
    showDownloadDialog();

    // 下载文件
    OkHttpUtils
            .get()
            .url(apkUrl)
            .build()
            .execute(new FileCallBack(apkPath, apkName) {

                @Override
                public void onResponse(File arg0, int arg1) {
                    mDownloadDialog.dismiss();
                    // 下载完成安装apk
                    installAPK(arg0.getAbsolutePath());
                }

                @Override
                public void onError(Call arg0, Exception arg1, int arg2) {
                    mDownloadDialog.dismiss();
                    ProgressHUD.showInfo(mContext, "您的网络不给力哦");
                }

                @Override
                public void inProgress(float progress, long total, int id) {
                    // 更新下载进度
                    mDownloadDialog.setProgress(Math.round(progress * 100));
                }

            });

}

/**
 * 弹出下载对话框
 */
public void showDownloadDialog() {
    mDownloadDialog = new ProgressDialog(mContext);
    mDownloadDialog.setIcon(R.mipmap.ic_launcher);
    mDownloadDialog.setTitle("版本更新");
    mDownloadDialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
    mDownloadDialog.setMessage("正在玩命下载中......");
    mDownloadDialog.getWindow().setGravity(Gravity.CENTER);
    mDownloadDialog.setMax(100);
    mDownloadDialog.show();
}

/**
 * 安装下载的新版本apk
 *
 * @param apkPath apk存放路径
 */
private void installAPK(String apkPath) {
    Intent intent = new Intent();
    intent.setAction("android.intent.action.VIEW");
    intent.addCategory("android.intent.category.DEFAULT");
    intent.setDataAndType(Uri.fromFile(new File(apkPath)), "application/vnd.android.package-archive");
    startActivity(intent);
}