Android應用自動更新功能的代碼實現
來源:互聯網
上載者:User
由於Android項目開源所致,市面上出現了N多安卓軟體市場。為了讓我們開發的軟體有更多的使用者使用,我們需要向N多市場發布,軟體升級後,我們也必須到安卓市場上進行更新,給我們增加了工作量。因此我們有必要給我們的Android應用增加自動更新的功能。既然實現自動更新,我們首先必須讓我們的應用知道是否存在新版本的軟體,因此我們可以在自己的網站上放置設定檔,存放軟體的版本資訊:<update> <version>2</version> <name>baidu_xinwen_1.1.0</name> <url>http://gdown.baidu.com/data/wisegame/f98d235e39e29031/baiduxinwen.apk</url></update>在這裡我使用的是XML檔案,方便讀取。由於XML檔案內容比較少,因此可通過DOM方式進行檔案的解析:public class ParseXmlService{ public HashMap<String, String> parseXml(InputStream inStream) throws Exception { HashMap<String, String> hashMap = new HashMap<String, String>(); // 執行個體化一個文檔構建器工廠 DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); // 通過文檔構建器工廠擷取一個文檔構建器 DocumentBuilder builder = factory.newDocumentBuilder(); // 通過文檔通過文檔構建器構建一個文檔執行個體 Document document = builder.parse(inStream); //擷取XML檔案根節點 Element root = document.getDocumentElement(); //獲得所有子節點 NodeList childNodes = root.getChildNodes(); for (int j = 0; j < childNodes.getLength(); j++) { //遍曆子節點 Node childNode = (Node) childNodes.item(j); if (childNode.getNodeType() == Node.ELEMENT_NODE) { Element childElement = (Element) childNode; //版本號碼 if ("version".equals(childElement.getNodeName())) { hashMap.put("version",childElement.getFirstChild().getNodeValue()); } //軟體名稱 else if (("name".equals(childElement.getNodeName()))) { hashMap.put("name",childElement.getFirstChild().getNodeValue()); } // else if (("url".equals(childElement.getNodeName()))) { hashMap.put("url",childElement.getFirstChild().getNodeValue()); } } } return hashMap; }}通過parseXml()方法,我們可以擷取伺服器上應用的版本、檔案名稱以及。緊接著我們就需要擷取到我們手機上應用的版本資訊:/** * 擷取軟體版本號碼 * * @param context * @return */private int getVersionCode(Context context){ int versionCode = 0; try { // 擷取軟體版本號碼, versionCode = context.getPackageManager().getPackageInfo("com.szy.update", 0).versionCode; } catch (NameNotFoundException e) { e.printStackTrace(); } return versionCode;}通過該方法我們擷取到的versionCode對應AndroidManifest.xml下android:versionCode。android:versionCode和android:versionName兩個屬性分別表示版本號碼,版本名稱。versionCode是整數型,而versionName是字串。由於versionName是給使用者看的,不太容易比較大小,升級檢查時,就可以檢查versionCode。把擷取到的手機上應用版本與伺服器端的版本進行比較,應用就可以判斷處是否需要更新軟體。處理流程處理代碼package com.szy.update; import java.io.File;import java.io.FileOutputStream;import java.io.IOException;import java.io.InputStream;import java.net.HttpURLConnection;import java.net.MalformedURLException;import java.net.URL;import java.util.HashMap; import android.app.AlertDialog;import android.app.Dialog;import android.app.AlertDialog.Builder;import android.content.Context;import android.content.DialogInterface;import android.content.Intent;import android.content.DialogInterface.OnClickListener;import android.content.pm.PackageManager.NameNotFoundException;import android.net.Uri;import android.os.Environment;import android.os.Handler;import android.os.Message;import android.view.LayoutInflater;import android.view.View;import android.widget.ProgressBar;import android.widget.Toast; /** *@author coolszy *@date 2012-4-26 *@blog http://blog.92coding.com */ public class UpdateManager{ /* 下載中 */ private static final int DOWNLOAD = 1; /* 下載結束 */ private static final int DOWNLOAD_FINISH = 2; /* 儲存解析的XML資訊 */ HashMap<String, String> mHashMap; /* 下載儲存路徑 */ private String mSavePath; /* 記錄進度條數量 */ private int progress; /* 是否取消更新 */ private boolean cancelUpdate = false; private Context mContext; /* 更新進度條 */ private ProgressBar mProgress; private Dialog mDownloadDialog; private Handler mHandler = new Handler() { public void handleMessage(Message msg) { switch (msg.what) { // 正在下載 case DOWNLOAD: // 設定進度條位置 mProgress.setProgress(progress); break; case DOWNLOAD_FINISH: // 安裝檔案 installApk(); break; default: break; } }; }; public UpdateManager(Context context) { this.mContext = context; } /** * 檢測軟體更新 */ public void checkUpdate() { if (isUpdate()) { // 顯示提示對話方塊 showNoticeDialog(); } else { Toast.makeText(mContext, R.string.soft_update_no, Toast.LENGTH_LONG).show(); } } /** * 檢查軟體是否有更新版本 * * @return */ private boolean isUpdate() { // 擷取當前軟體版本 int versionCode = getVersionCode(mContext); // 把version.xml放到網路上,然後擷取檔案資訊 InputStream inStream = ParseXmlService.class.getClassLoader().getResourceAsStream("version.xml"); // 解析XML檔案。 由於XML檔案比較小,因此使用DOM方式進行解析 ParseXmlService service = new ParseXmlService(); try { mHashMap = service.parseXml(inStream); } catch (Exception e) { e.printStackTrace(); } if (null != mHashMap) { int serviceCode = Integer.valueOf(mHashMap.get("version")); // 版本判斷 if (serviceCode > versionCode) { return true; } } return false; } /** * 擷取軟體版本號碼 * * @param context * @return */private int getVersionCode(Context context){ int versionCode = 0; try { // 擷取軟體版本號碼,對應AndroidManifest.xml下android:versionCode versionCode = context.getPackageManager().getPackageInfo("com.szy.update", 0).versionCode; } catch (NameNotFoundException e) { e.printStackTrace(); } return versionCode;} /** * 顯示軟體更新對話方塊 */ private void showNoticeDialog() { // 構造對話方塊 AlertDialog.Builder builder = new Builder(mContext); builder.setTitle(R.string.soft_update_title); builder.setMessage(R.string.soft_update_info); // 更新 builder.setPositiveButton(R.string.soft_update_updatebtn, new OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { dialog.dismiss(); // 顯示下載對話方塊 showDownloadDialog(); } }); // 稍後更新 builder.setNegativeButton(R.string.soft_update_later, new OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { dialog.dismiss(); } }); Dialog noticeDialog = builder.create(); noticeDialog.show(); } /** * 顯示軟體下載對話方塊 */ private void showDownloadDialog() { // 構造軟體下載對話方塊 AlertDialog.Builder builder = new Builder(mContext); builder.setTitle(R.string.soft_updating); // 給下載對話方塊增加進度條 final LayoutInflater inflater = LayoutInflater.from(mContext); View v = inflater.inflate(R.layout.softupdate_progress, null); mProgress = (ProgressBar) v.findViewById(R.id.update_progress); builder.setView(v); // 取消更新 builder.setNegativeButton(R.string.soft_update_cancel, new OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { dialog.dismiss(); // 設定取消狀態 cancelUpdate = true; } }); mDownloadDialog = builder.create(); mDownloadDialog.show(); // 現在檔案 downloadApk(); } /** * 下載apk檔案 */ private void downloadApk() { // 啟動新線程下載軟體 new downloadApkThread().start(); } /** * 下載檔案線程 * * @author coolszy *@date 2012-4-26 *@blog http://blog.92coding.com */ private class downloadApkThread extends Thread { @Override public void run() { try { // 判斷SD卡是否存在,並且是否具有讀寫權限 if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) { // 獲得儲存卡的路徑 String sdpath = Environment.getExternalStorageDirectory() + "/"; mSavePath = sdpath + "download"; URL url = new URL(mHashMap.get("url")); // 建立串連 HttpURLConnection conn = (HttpURLConnection) url.openConnection(); conn.connect(); // 擷取檔案大小 int length = conn.getContentLength(); // 建立輸入資料流 InputStream is = conn.getInputStream(); File file = new File(mSavePath); // 判斷檔案目錄是否存在 if (!file.exists()) { file.mkdir(); } File apkFile = new File(mSavePath, mHashMap.get("name")); FileOutputStream fos = new FileOutputStream(apkFile); int count = 0; // 緩衝 byte buf[] = new byte[1024]; // 寫入到檔案中 do { int numread = is.read(buf); count += numread; // 計算進度條位置 progress = (int) (((float) count / length) * 100); // 更新進度 mHandler.sendEmptyMessage(DOWNLOAD); if (numread <= 0) { // 下載完成 mHandler.sendEmptyMessage(DOWNLOAD_FINISH); break; } // 寫入檔案 fos.write(buf, 0, numread); } while (!cancelUpdate);// 點擊取消就停止下載. fos.close(); is.close(); } } catch (MalformedURLException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } // 取消下載對話方塊顯示 mDownloadDialog.dismiss(); } }; /** * 安裝APK檔案 */ private void installApk() { File apkfile = new File(mSavePath, mHashMap.get("name")); if (!apkfile.exists()) { return; } // 通過Intent安裝APK檔案 Intent i = new Intent(Intent.ACTION_VIEW); i.setDataAndType(Uri.parse("file://" + apkfile.toString()), "application/vnd.android.package-archive"); mContext.startActivity(i); }}