Android软件自动更新

时间:2021-11-22 22:52:53

所有好的手机应用程序都会有这项功能实现。所以我想做一个工具类UpdateManager.java ,然后在activity中直接调用方法 checkUpdate() 检测是否有更新。这样就可以一劳永逸了O(∩_∩)O!

先说说软件自动更新做的好处:

1、开发者不需要每次都去各个市场平台发布新版本的软件,可以省去很多时间,金钱;

2、用户不需要去关注软件是否有更新,可以提高用户满意度。

再说说其实现原理,这里以流程图展现给大家:

Android软件自动更新

看完之后,废话不多说,我们开始正式的开发了。

第一步,为了让软件知道最新的版本信息,我们需要在服务器端创建一个 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解析)代码如下:

代码里都有备注,欢迎有不懂或者有好的建议者留言交流。

Android软件自动更新Android软件自动更新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 }
Android软件自动更新Android软件自动更新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 inStream17      * @return18      * @throws Exception19      */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 }
Android软件自动更新Android软件自动更新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

 

 


<script type="text/javascript"><!--google_ad_client = "ca-pub-1944176156128447";/* cnblogs 首页横幅 */google_ad_slot = "5419468456";google_ad_width = 728;google_ad_height = 90;//--></script><script type="text/javascript" src="http://pagead2.googlesyndication.com/pagead/show_ads.js"></script>