简述更新的实现原理(想直奔主题的小伙伴可以略过这些):
服务器上存放要更新的 安装包、Json串,Json里主要是关于要更新内容的描述,例,
{
"url":"https://com.excaple/demo/test.apk", //相应的apk路径
"versionCode":2, //版本号,通过比较版本号判断是否需要更新
"updateMessage":"[1]新增***功能<br/>[2]优化了***<br/>[3]增强了稳定性" //此次更新的主要内容简介
}
app获取服务器上的Json串,解析得到url、versionCode、updateMessgae等相关信息,获取到app现在的VersionCode与Json串里相比较,如果大于(可以自己定义判断规则)就弹窗询问用户,用户同意就下载apk,然后进行以下操作。
分别针对两种安卓设备:1.已root 2.未root
一.已root的安卓设备(应用场景主要是一些特定的安卓设备,像广告机)
1.询问用户是否更新的弹窗是自己实现的,也就是说,可以不弹窗,直接下载。
2.下载完成后,进行静默安装,此方法不会弹窗提示用户
静默安装是需要root权限的,直接上方法吧
/*
@pararm apkPath 等待安装的app全路径,如:/sdcard/app/app.apk
**/
private static boolean clientInstall(String apkPath) {
PrintWriter PrintWriter = null;
Process process = null;
try {
process = Runtime.getRuntime().exec("su");
PrintWriter = new PrintWriter(process.getOutputStream());
PrintWriter.println("chmod 777 " + apkPath);
PrintWriter
.println("export LD_LIBRARY_PATH=/vendor/lib:/system/lib");
PrintWriter.println("pm install -r " + apkPath);
// PrintWriter.println("exit");
PrintWriter.flush();
PrintWriter.close();
int value = process.waitFor();
Log.d("TAG","静默安装返回值:"+value);
return true;
} catch (Exception e) {
e.printStackTrace();
Log.d("TAG","安装apk出现异常");
} finally {
if (process != null) {
process.destroy();
}
}
return false;
}
3.安装完成后自动打开的方法:
在软件升级、安装、卸载时,系统会分别发这三个广播:
<action android:name="android.intent.action.PACKAGE_REPLACED"/>
<action android:name="android.intent.action.PACKAGE_ADDED" />
<action android:name="android.intent.action.PACKAGE_REMOVED" />
所以我们根据自己需要可以分别接收这几个广播,我做的是升级,所以接收 "PACKAGE_REPLACED" 这个广播,
接收到广播后要判断软件包名是否一致。如果一致则自启动软件,下面是我接收的广播,
public class UpdateReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
if (intent.getAction().equals("android.intent.action.PACKAGE_REPLACED")){
if (intent.getDataString().equals(context.getPackageName())){
Log.d("TAG","收到广播,包名一致");
Intent intent2 = new Intent(context, MainActivity.class);
intent2.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent2);
}else {
Log.d("TAG","收到广播,包名不一致");
}
}
}
}
如果不判断包名是否一致,有别的软件安装或升级,而你是接收了这个广播的,只要接收了这个广播的都会启动,
所以会造成启动混乱。这时候你用 intent.getDataString(),取到传过来的包名与自己的包名进行一下比较一下,
符合条件启动就OK了。
注册广播:
<receiver android:name=".receiver.UpdateReceiver">
<intent-filter>
<action android:name="android.intent.action.PACKAGE_REPLACED"/>
<data android:scheme="package"/>
</intent-filter>
</receiver>
如果你的使用场景不是升级自身,而是安装别的apk并打开,你监听 "PACKAGE_ADDED"这个就可以实现了。