Android軟件自動更新


  所有好的手機應用程序都會有這項功能實現。所以我想做一個工具類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解析)代碼如下:

代碼里都有備注,歡迎有不懂或者有好的建議者留言交流。

HttpUtil.java
  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 }
ParseXmlService.java
 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 }
UpdateManager.java
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

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM