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,這樣的使用體驗更加平滑。