所有好的手機應用程序都會有這項功能實現。所以我想做一個工具類UpdateManager.java ,然后在activity中直接調用方法 checkUpdate() 檢測是否有更新。這樣就可以一勞永逸了O(∩_∩)O!
先說說軟件自動更新做的好處:
1、開發者不需要每次都去各個市場平台發布新版本的軟件,可以省去很多時間,金錢;
2、用戶不需要去關注軟件是否有更新,可以提高用戶滿意度。
再說說其實現原理,這里以流程圖展現給大家:
看完之后,廢話不多說,我們開始正式的開發了。
第一步,為了讓軟件知道最新的版本信息,我們需要在服務器端創建一個 app_version.xml 文件,放在服務器下,用於存放軟件版本信息,代碼如下:
1 <app> 2 <version>130</version>軟件版本號 3 <name>appname_1.3.0</name>軟件此版本的名稱(這里如果不加后綴名“.apk”,則一定要在客戶端中下載軟件時加上) 4 <url>http://.......</url>軟件下載地址 5 </app>
下面信息表示軟件的版本信息,想信大家知道判斷軟件是否一樣,比較的就是版本號 versionCode 吧。
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.tq365.android.activity" android:versionCode="120" android:versionName="1.2.0" >
第二步,創建工具類 UpdateManager.java ,負責軟件更新功能模塊,其中還用到了,兩個自編工具類 HttpUtil.java(網絡服務)、ParseXmlService.java(XML解析)代碼如下:
代碼里都有備注,歡迎有不懂或者有好的建議者留言交流。

1 import java.io.IOException; 2 import java.io.InputStream; 3 import java.io.Serializable; 4 import java.lang.reflect.Field; 5 import java.util.ArrayList; 6 import java.util.List; 7 8 import org.apache.http.HttpEntity; 9 import org.apache.http.HttpResponse; 10 import org.apache.http.NameValuePair; 11 import org.apache.http.ParseException; 12 import org.apache.http.client.HttpClient; 13 import org.apache.http.client.entity.UrlEncodedFormEntity; 14 import org.apache.http.client.methods.HttpGet; 15 import org.apache.http.client.methods.HttpPost; 16 import org.apache.http.client.methods.HttpUriRequest; 17 import org.apache.http.impl.client.DefaultHttpClient; 18 import org.apache.http.message.BasicNameValuePair; 19 import org.apache.http.util.EntityUtils; 20 21 import android.util.Log; 22 23 public class HttpUtil { 24 private static final String TAG = "HttpUtil"; 25 26 public static final int METHOD_GET = 1; 27 public static final int METHOD_POST = 2; 28 public static final String BASE_URL = "http://...."; 29 30 /** 31 * 遠程訪問服務器 32 * 33 * @param uri 34 * @param params 35 * 參數 36 * @param method 37 * 訪問方式get/post 38 * @return HttpEntity 39 * @throws IOException 40 */ 41 public static HttpEntity getEntity(String uri, 42 ArrayList<BasicNameValuePair> params, int method) 43 throws IOException { 44 HttpEntity entity = null; 45 HttpClient client = new DefaultHttpClient(); 46 HttpUriRequest request = null; 47 switch (method) { 48 case METHOD_GET: 49 StringBuffer sb = null; 50 if (uri.indexOf("http")==0) { 51 sb = new StringBuffer(uri); 52 }else { 53 sb = new StringBuffer(BASE_URL + uri); 54 } 55 if (params != null && !params.isEmpty()) { 56 sb.append("?"); 57 for (BasicNameValuePair param : params) { 58 sb.append(param.getName()).append("=") 59 .append(param.getValue()).append("&"); 60 } 61 sb.deleteCharAt(sb.length() - 1); 62 } 63 request = new HttpGet(sb.toString()); 64 break; 65 case METHOD_POST: 66 if (uri.indexOf("http")==0) { 67 request = new HttpPost(uri); 68 }else { 69 request = new HttpPost(BASE_URL + uri); 70 } 71 if (params != null && !params.isEmpty()) { 72 UrlEncodedFormEntity reqEntity = new UrlEncodedFormEntity(params,"UTF-8"); 73 ((HttpPost) request).setEntity(reqEntity); 74 } 75 break; 76 } 77 HttpResponse response = client.execute(request); 78 if (response.getStatusLine().getStatusCode() == 200) { 79 entity = response.getEntity(); 80 } 81 return entity; 82 } 83 84 /** 85 * 訪問服務器,返回IO流 86 * 87 * @param uri 88 * @param params 89 * @param method 90 * @return InputStream 91 * @throws IOException 92 */ 93 public static InputStream getInputStream(String uri, 94 ArrayList<BasicNameValuePair> params, int method) 95 throws IOException { 96 HttpEntity httpEntity = getEntity(uri, params, method); 97 if (httpEntity == null) { 98 return null; 99 } 100 return httpEntity.getContent(); 101 } 102 103 }

1 import java.io.InputStream; 2 import java.util.HashMap; 3 4 import javax.xml.parsers.DocumentBuilder; 5 import javax.xml.parsers.DocumentBuilderFactory; 6 7 import org.w3c.dom.Document; 8 import org.w3c.dom.Element; 9 import org.w3c.dom.Node; 10 import org.w3c.dom.NodeList; 11 12 public class ParseXmlService { 13 14 /** 15 * 解析XML,軟件版本信息 16 * @param inStream 17 * @return 18 * @throws Exception 19 */ 20 public HashMap<String, String> parseXml(InputStream inStream) 21 throws Exception { 22 HashMap<String, String> hashMap = new HashMap<String, String>(); 23 24 // 實例化一個文檔構建器工廠 25 DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); 26 // 通過文檔構建器工廠獲取一個文檔構建器 27 DocumentBuilder builder = factory.newDocumentBuilder(); 28 // 通過文檔通過文檔構建器構建一個文檔實例 29 Document document = builder.parse(inStream); 30 // 獲取XML文件根節點 31 Element root = document.getDocumentElement(); 32 // 獲得所有子節點 33 NodeList childNodes = root.getChildNodes(); 34 for (int j = 0; j < childNodes.getLength(); j++) { 35 // 遍歷子節點 36 Node childNode = (Node) childNodes.item(j); 37 if (childNode.getNodeType() == Node.ELEMENT_NODE) { 38 Element childElement = (Element) childNode; 39 // 版本號 40 if ("version".equals(childElement.getNodeName())) { 41 hashMap.put("version", childElement.getFirstChild().getNodeValue()); 42 } 43 // 軟件名稱 44 else if (("name".equals(childElement.getNodeName()))) { 45 hashMap.put("name", childElement.getFirstChild().getNodeValue()); 46 } 47 // 下載地址 48 else if (("url".equals(childElement.getNodeName()))) { 49 hashMap.put("url", childElement.getFirstChild().getNodeValue()); 50 } 51 } 52 } 53 return hashMap; 54 } 55 }

import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.net.HttpURLConnection; import java.net.URL; import java.util.HashMap; import com.tq365.android.activity.R; import android.app.AlertDialog; import android.app.AlertDialog.Builder; import android.app.Dialog; 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; public class UpdateManager { private Context mContext; // 更新進度條 private ProgressBar mUpdateProgressBar; // 記錄進度條數量 private int progress; // 保存解析的XML信息 private HashMap<String, String> mHashMap; // 是否取消更新 private boolean cancelUpdate = false; // 下載狀態--下載中 private static final int DOWNLOAD_ING = 1; // 下載狀態--下載成功 private static final int DOWNLOAD_SUCCESS = 2; // 下載狀態--下載失敗 private static final int DOWNLOAD_FAIL = 3; // 下載保存路徑 private String mSavePath; //下載對話框 private Dialog mDownloadDialog; private Handler mHandler; private boolean isLoop; public UpdateManager(Context context) { super(); this.mContext = context; mHandler = new Handler(){ @Override public void handleMessage(Message msg) { switch (msg.what) { case DOWNLOAD_ING: mUpdateProgressBar.setProgress(progress); break; case DOWNLOAD_SUCCESS: //安裝APK installApk(); break; case DOWNLOAD_FAIL: Toast.makeText(mContext, "文件下載失敗!", Toast.LENGTH_LONG).show(); break; } } }; } /** * 檢測軟件更新 * * @param isAuto * 為true:軟件自動檢測更新;false:用戶手動檢測更新。 */ public void checkUpdate(boolean isAuto) { try { if (isUpdate()) { // 顯示提示對話框 showNoticeDialog(); } else if (!isAuto) { // 告訴用戶已是最新版本,不需要更新。 Toast.makeText(mContext, "您的軟件已是最新版本,不需要更新!", Toast.LENGTH_SHORT).show(); } } catch (Exception e) { e.printStackTrace(); } } /** * 檢測軟件是否有更新 * * @return * @throws Exception */ private boolean isUpdate() throws Exception { //獲取當前軟件版本 int versionCode = getVersionVode(mContext); //獲取我們之前放在服務器端的app_version.xml的文件信息 //Android 3.0(含)之后訪問網絡都不能在主線程中 isLoop = true; new Thread(){ @Override public void run() { try { InputStream inStream = HttpUtil.getInputStream("apks/app_version.xml", null, HttpUtil.METHOD_GET); //解析xml文件。由於XML文件較小,我們采用DOM方式進行解析 ParseXmlService service = new ParseXmlService();//這個類是自己寫的解析XML的工具類 try { mHashMap = service.parseXml(inStream); } catch (Exception e) { e.printStackTrace(); }finally{ isLoop = false; } } catch (IOException e) { e.printStackTrace(); } } }.start(); while (isLoop) { Thread.sleep(1000); } if (null != mHashMap) { int serviceCode = Integer.valueOf(mHashMap.get("version")); //判斷版本號 if (serviceCode > versionCode) { return true; } } return false; } /** * 獲取軟件當前版本號 * * @param context * @return */ private int getVersionVode(Context context){ int versionCode = 0; // 獲取軟件版本號,對應AndroidManifest.xml下android:versionCode try { versionCode = context.getPackageManager().getPackageInfo("com.tq365.android.activity", 0).versionCode; } catch (NameNotFoundException e) { e.printStackTrace(); } return versionCode; } /** * 顯示軟件更新對話框 */ private void showNoticeDialog() { AlertDialog.Builder builder = new Builder(mContext); builder.setTitle("軟件更新") .setMessage("軟件有新版本,要更新嗎?") .setPositiveButton("立即更新", new OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { dialog.dismiss(); //顯示軟件下載對話框 showDownloadDialog(); } }) .setNegativeButton("稍后再說", new OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { dialog.dismiss(); } }).create().show(); } /** * 顯示軟件下載對話框 */ private void showDownloadDialog() { AlertDialog.Builder builder = new Builder(mContext); builder.setTitle("正在更新"); //給對話框增加進度條 LayoutInflater inflater = LayoutInflater.from(mContext); View v = inflater.inflate(R.layout.update_progress, null); mUpdateProgressBar = (ProgressBar) v.findViewById(R.id.update_progressBar); mDownloadDialog = builder.setView(v) .setNegativeButton("取消更新", new OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { dialog.dismiss(); //設置取消狀態 cancelUpdate = true; } }).create(); mDownloadDialog.show(); //下載APK文件 downloadApk(); } /** * 下載APK文件 */ private void downloadApk() { //啟動下載APK線程 new DownloadApkThread().start(); } /** * 下載APK文件線程 */ private class DownloadApkThread extends Thread { @Override public void run() { try { // 判斷SD卡是否存在,並且是否有讀寫權限 if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) { // 獲取SD卡路徑 String sdPadth = Environment.getExternalStorageDirectory()+ ""; mSavePath = sdPadth + "/download"; URL url = new URL(mHashMap.get("url")); // 創建連接 HttpURLConnection conn = (HttpURLConnection) url.openConnection(); conn.connect(); // 獲取文件大小 int fileSize = conn.getContentLength(); // 創建輸入流 InputStream inStream = 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[] b = new byte[1024]; do { int numRead = inStream.read(b); count += numRead; // 計算進度條位置 progress = (int) (((float) count / fileSize) * 100); // 更新進度 mHandler.sendEmptyMessage(DOWNLOAD_ING); if (numRead <= 0) {// 下載完成 mHandler.sendEmptyMessage(DOWNLOAD_SUCCESS); break; } // 寫入文件 fos.write(b, 0, numRead); } while (!cancelUpdate); } } catch (Exception e) { mHandler.sendEmptyMessage(DOWNLOAD_FAIL); } finally { mDownloadDialog.dismiss(); } } } /** * 安裝APK */ private void installApk() { File apk = new File(mSavePath,mHashMap.get("name")); if (!apk.exists()) { return; } //通過Intent安裝APK文件 Intent intent = new Intent(Intent.ACTION_VIEW); intent.setDataAndType(Uri.parse("file://"+apk.toString()), "application/vnd.android.package-archive"); mContext.startActivity(intent); } }
如果轉載請尊重作者的勞動果實,附上http://www.cnblogs.com/small-bai/archive/2013/03/12/2955852.html。