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

 

 

posted @ 2013-03-15 09:28  三两土豆  阅读(2194)  评论(8编辑  收藏  举报