Android 應用強制更新策略

Android應用強制更新的用途十分廣泛,特別上剛上線的應用肯定會存在或多或少的bug,特別是涉及移動支付這一塊的內容,如果出錯了會造成比較大的損失,所以強制更新顯得尤爲重要。

一般來說,強制更新的策略就是:
應用啓動時請求後臺,後臺發送應用最新版本的信息(包括應用版本號、名稱、更新內容說明、下載包的服務器地址、是否強制更新的標誌位)等等。

下面我們就將根據以上思路來寫實現代碼。

1.AndroidManifest配置版本信息
在AndroidManifest.xml裏定義了每個Android apk的版本標識:

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.demo"
android:versionCode="1"
android:versionName="1.0.0">
<application>
</application>
</manifest>

其中,package=”com.demo”是我們的包名;android:versionCode=”1”是版本代號,爲整型數字;android:versionName=”1.0.0”是版本名稱,爲字符串,顯示給用戶看的。

當需要讀取AndroidManifest文件中版本號和版本名稱的時候,用packageManager就可以簡單得到。代碼如下所示:

public static int getVerCode(Context context) {
int verCode = -1;
try {
verCode = context.getPackageManager().getPackageInfo(
"com.demo", 0).versionCode;
} catch (NameNotFoundException e) {
Log.e(TAG, e.getMessage());
}
return verCode;
}
public static String getVerName(Context context) {
String verName = "";
try {
verName = context.getPackageManager().getPackageInfo(
"com.demo", 0).versionName;
} catch (NameNotFoundException e) {
Log.e(TAG, e.getMessage());
}
return verName;
}

2.進行版本檢查
在服務端放置最新版本的apk文件,如:http://localhost/mydemo/demo.apk
同時,在服務端放置對應此apk的版本信息調用接口或者文件,如:http://localhost/mydemo/ver.json
ver.json中的內容爲:
[{“appname”:”jtapp12”,”apkname”:”jtapp-12-updateapksamples.apk”,”verName”:1.0.1,”verCode”:2}]
然後,在手機客戶端上進行版本讀取和檢查:

private boolean getServerVer () {
try {
String verjson = NetworkTool.getContent(Config.UPDATE_SERVER
+ Config.UPDATE_VERJSON);
JSONArray array = new JSONArray(verjson);
if (array.length() > 0) {
JSONObject obj = array.getJSONObject(0);
try {
newVerCode = Integer.parseInt(obj.getString("verCode"));
newVerName = obj.getString("verName");
} catch (Exception e) {
newVerCode = -1;
newVerName = "";
return false;
}
}
} catch (Exception e) {
Log.e(TAG, e.getMessage());
return false;
}
return true;
}

比較服務器和客戶端的版本,並進行更新操作。

if (getServerVerCode()) {
int vercode = Config.getVerCode(this); // 用到前面第一節寫的方法
if (newVerCode > vercode) {
doNewVersionUpdate(); // 更新新版本
} else {
notNewVersionShow(); // 提示當前爲最新版本
}
}

調用方法:

private void notNewVersionShow() {
int verCode = Config.getVerCode(this);
String verName = Config.getVerName(this);
StringBuffer sb = new StringBuffer();
sb.append("當前版本:");
sb.append(verName);
sb.append(" Code:");
sb.append(verCode);
sb.append(",\n已是最新版,無需更新!");
Dialog dialog = new AlertDialog.Builder(Update.this).setTitle("軟件更新")
.setMessage(sb.toString())// 設置內容
.setPositiveButton("確定",// 設置確定按鈕
new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog,
int which) {
finish();
}
}).create();// 創建
// 顯示對話框
dialog.show();
}
private void doNewVersionUpdate() {
int verCode = Config.getVerCode(this);
String verName = Config.getVerName(this);
StringBuffer sb = new StringBuffer();
sb.append("當前版本:");
sb.append(verName);
sb.append(" Code:");
sb.append(verCode);
sb.append(", 發現新版本:");
sb.append(newVerName);
sb.append(" Code:");
sb.append(newVerCode);
sb.append(", 是否更新?");
Dialog dialog = new AlertDialog.Builder(Update.this)
.setTitle("軟件更新")
.setMessage(sb.toString())
// 設置內容
.setPositiveButton("更新",// 設置確定按鈕
new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog,
int which) {
pBar = new ProgressDialog(Update.this);
pBar.setTitle("正在下載");
pBar.setMessage("請稍候...");
pBar.setProgressStyle(ProgressDialog.STYLE_SPINNER);
downFile(Config.UPDATE_SERVER + Config.UPDATE_APKNAME);
}
})
.setNegativeButton("暫不更新",
new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog,
int whichButton) {
// 點擊"取消"按鈕之後退出程序
finish();
}
}).create();// 創建
// 顯示對話框
dialog.show();
}

更新下載:

void downFile(final String url) {
pBar.show();
new Thread() {
public void run() {
HttpClient client = new DefaultHttpClient();
HttpGet get = new HttpGet(url);
HttpResponse response;
try {
response = client.execute(get);
HttpEntity entity = response.getEntity();
long length = entity.getContentLength();
InputStream is = entity.getContent();
FileOutputStream fileOutputStream = null;
if (is != null) {
File file = new File(
Environment.getExternalStorageDirectory(),
Config.UPDATE_SAVENAME);
fileOutputStream = new FileOutputStream(file);
byte[] buf = new byte[1024];
int ch = -1;
int count = 0;
while ((ch = is.read(buf)) != -1) {
fileOutputStream.write(buf, 0, ch);
count += ch;
if (length > 0) {
}
}
}
fileOutputStream.flush();
if (fileOutputStream != null) {
fileOutputStream.close();
}
down();
} catch (ClientProtocolException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}.start();
}

下載完成,通過handler通知主ui線程將下載對話框取消。

void down() {
handler.post(new Runnable() {
public void run() {
pBar.cancel();
update();
}
});
}

安裝應用

void update() {
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setDataAndType(Uri.fromFile(new File(Environment
.getExternalStorageDirectory(), Config.UPDATE_SAVENAME)),
"application/vnd.android.package-archive");
startActivity(intent);
}

如果你將apk應用發佈到market上,那麼,你會發現market內建了類似的模塊,可以自動更新或者提醒你是否更新應用。那麼,對於你自己的應用需要自動更新的話,自己內建一個是不是更加方便了呢?本文提到的代碼大多是在UpdateActivity.java中實現,爲了能夠使更新過程更加友好,可以在最初launcher的Activity中建立一個線程,用來檢查服務端是否有更新。有更新的時候就啓動UpdateActivity,這樣的使用體驗更加平滑。

(待完成)

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章