• Android


    转自:http://blog.csdn.net/wwj_748/article/details/8195565

    接触到一个很实用的技术,那就是软件自动更新。一般开发者是通过自行在应用平台添加更新版本的apk。这样做,如果是在一两个应用平台发布应用,那还说得过去,工作量还不是很大。但大家都知道,Android开发者是比较苦逼的。由于开源所致,出现了N多应用市场。如果想赚取更多的收入,那就要在各个应用市场进行更新。那就悲催咯。

    比较出名的一些应用市场有如下:

     

                                                                    

    那如何实现软件自动更新,下面是具体实例:

    效果图:

       

        

    具体步骤:

    1. 在服务器上部署更新所用的xml文件:version.xml

    [html] view plaincopy
     
    1. <update>  
    2.     <version>2</version>  
    3.     <name>baiduxinwen.apk</name>  
    4.     <url>http://gdown.baidu.com/data/wisegame/e5f5c3b8e59401c8/baiduxinwen.apk</url>  
    5. </update>  


     

    2. 在客户端实现更新操作

      涉及到三个技术:

       1.xml文件的解析

       2.HttpURLConnection连接

       3.文件流I/O

     

    这里创建一个解析xml文件的服务类:ParXmlService.java

    [java] view plaincopy
     
    1. package com.xiaowu.news.update;  
    2.   
    3. import java.io.InputStream;  
    4. import java.util.HashMap;  
    5.   
    6. import javax.xml.parsers.DocumentBuilder;  
    7. import javax.xml.parsers.DocumentBuilderFactory;  
    8.   
    9. import org.w3c.dom.Document;  
    10. import org.w3c.dom.Element;  
    11. import org.w3c.dom.Node;  
    12. import org.w3c.dom.NodeList;  
    13.   
    14. public class ParseXmlService {  
    15.     public HashMap<String, String> parseXml (InputStream inStream) throws Exception{  
    16.         HashMap<String, String> hashMap = new HashMap<String, String>();  
    17.         //创建DocumentBuilderFactory,该对象将创建DocumentBuilder。  
    18.         DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();  
    19.         //创建DocumentBuilder,DocumentBuilder将实际进行解析以创建Document对象  
    20.         DocumentBuilder builder = factory.newDocumentBuilder();  
    21.         //解析该文件以创建Document对象  
    22.         Document document = builder.parse(inStream);  
    23.         //获取XML文件根节点   
    24.         Element root = document.getDocumentElement();  
    25.         //获得所有子节点  
    26.         NodeList childNodes = root.getChildNodes();  
    27.         for(int i = 0; i < childNodes.getLength(); i++) {  
    28.             Node childNode = (Node) childNodes.item(i);  
    29.             if(childNode.getNodeType() == Node.ELEMENT_NODE) {  
    30.                 Element childElement = (Element) childNode;  
    31.                 //版本号   
    32.                 if("version".equals(childElement.getNodeName())) {  
    33.                     hashMap.put("version", childElement.getFirstChild().getNodeValue());  
    34.                 //软件名称   
    35.                 } else if("name".equals(childElement.getNodeName())) {  
    36.                     hashMap.put("name", childElement.getFirstChild().getNodeValue());  
    37.                 //下载地址  
    38.                 } else if("url".equals(childElement.getNodeName())) {  
    39.                     hashMap.put("url", childElement.getFirstChild().getNodeValue());  
    40.                 }  
    41.             }  
    42.               
    43.         }  
    44.         return hashMap;  
    45.     }  
    46. }  


    实现更新操作的管理类:UpdateManager.java

    [java] view plaincopy
     
      1. package com.xiaowu.news.update;  
      2.   
      3. import java.io.File;  
      4. import java.io.FileOutputStream;  
      5. import java.io.IOException;  
      6. import java.io.InputStream;  
      7. import java.net.HttpURLConnection;  
      8. import java.net.MalformedURLException;  
      9. import java.net.URL;  
      10. import java.util.HashMap;  
      11.   
      12. import javax.net.ssl.HttpsURLConnection;  
      13.   
      14. import android.app.AlertDialog;  
      15. import android.app.AlertDialog.Builder;  
      16. import android.app.Dialog;  
      17. import android.content.Context;  
      18. import android.content.DialogInterface;  
      19. import android.content.Intent;  
      20. import android.content.DialogInterface.OnClickListener;  
      21. import android.content.pm.PackageManager.NameNotFoundException;  
      22. import android.net.Uri;  
      23. import android.os.Environment;  
      24. import android.os.Handler;  
      25. import android.view.LayoutInflater;  
      26. import android.view.View;  
      27. import android.widget.ProgressBar;  
      28. import android.widget.Toast;  
      29.   
      30. import com.xiaowu.news.R;  
      31.   
      32. /** 
      33.  *  
      34.  * @author wwj 
      35.  * @date 2012/11/17 
      36.  * 实现软件更新的管理类 
      37.  */  
      38. public class UpdateManager {  
      39.       
      40.     //下载中...  
      41.     private static final int DOWNLOAD = 1;  
      42.     //下载完成  
      43.     private static final int DOWNLOAD_FINISH = 2;  
      44.     //保存解析的XML信息  
      45.     HashMap<String , String> mHashMap;  
      46.     //下载保存路径  
      47.     private String mSavePath;  
      48.     //记录进度条数量  
      49.     private int progress;  
      50.     //是否取消更新  
      51.     private boolean cancelUpdate = false;  
      52.     //上下文对象  
      53.     private Context mContext;  
      54.     //进度条  
      55.     private ProgressBar mProgressBar;  
      56.     //更新进度条的对话框  
      57.     private Dialog mDownloadDialog;  
      58.       
      59.       
      60.     private Handler mHandler = new Handler() {  
      61.         public void handleMessage(android.os.Message msg) {  
      62.             switch(msg.what){  
      63.             //下载中。。。  
      64.             case DOWNLOAD:  
      65.                 //更新进度条  
      66.                 System.out.println(progress);  
      67.                 mProgressBar.setProgress(progress);  
      68.                 break;  
      69.             //下载完成  
      70.             case DOWNLOAD_FINISH:  
      71.                 // 安装文件  
      72.                 installApk();  
      73.                 break;  
      74.             }  
      75.         };  
      76.     };  
      77.   
      78.   
      79.     public UpdateManager(Context context) {  
      80.         super();  
      81.         this.mContext = context;  
      82.     }  
      83.       
      84.       
      85.     /** 
      86.      * 检测软件更新 
      87.      */  
      88.     public void checkUpdate() {  
      89.         if (isUpdate()) {  
      90.             //显示提示对话框  
      91.             showNoticeDialog();  
      92.         } else {  
      93.             Toast.makeText(mContext, R.string.soft_update_no, Toast.LENGTH_SHORT).show();  
      94.         }  
      95.           
      96.     }  
      97.       
      98.     private void showNoticeDialog() {  
      99.         // TODO Auto-generated method stub  
      100.         //构造对话框  
      101.         AlertDialog.Builder builder = new Builder(mContext);  
      102.         builder.setTitle(R.string.soft_update_title);  
      103.         builder.setMessage(R.string.soft_update_info);  
      104.         //更新  
      105.         builder.setPositiveButton(R.string.soft_update_updatebtn, new OnClickListener() {  
      106.               
      107.             @Override  
      108.             public void onClick(DialogInterface dialog, int which) {  
      109.                 // TODO Auto-generated method stub  
      110.                 dialog.dismiss();  
      111.                 // 显示下载对话框  
      112.                 showDownloadDialog();  
      113.             }  
      114.         });  
      115.         // 稍后更新  
      116.         builder.setNegativeButton(R.string.soft_update_later, new OnClickListener() {  
      117.               
      118.             @Override  
      119.             public void onClick(DialogInterface dialog, int which) {  
      120.                 // TODO Auto-generated method stub  
      121.                 dialog.dismiss();  
      122.             }  
      123.         });  
      124.         Dialog noticeDialog = builder.create();  
      125.         noticeDialog.show();  
      126.     }  
      127.       
      128.     private void showDownloadDialog() {  
      129.         // 构造软件下载对话框  
      130.         AlertDialog.Builder builder = new Builder(mContext);  
      131.         builder.setTitle(R.string.soft_updating);  
      132.         // 给下载对话框增加进度条  
      133.         final LayoutInflater inflater = LayoutInflater.from(mContext);  
      134.         View view = inflater.inflate(R.layout.softupdate_progress, null);  
      135.         mProgressBar = (ProgressBar) view.findViewById(R.id.update_progress);  
      136.         builder.setView(view);  
      137.         builder.setNegativeButton(R.string.soft_update_cancel, new OnClickListener() {  
      138.               
      139.             @Override  
      140.             public void onClick(DialogInterface dialog, int which) {  
      141.                 // TODO Auto-generated method stub  
      142.                 dialog.dismiss();  
      143.                 // 设置取消状态  
      144.                 cancelUpdate = true;  
      145.             }  
      146.         });  
      147.         mDownloadDialog = builder.create();  
      148.         mDownloadDialog.show();  
      149.         //下载文件  
      150.         downloadApk();  
      151.     }  
      152.       
      153.     /** 
      154.      * 下载APK文件 
      155.      */  
      156.     private void downloadApk() {  
      157.         // TODO Auto-generated method stub  
      158.         // 启动新线程下载软件  
      159.         new DownloadApkThread().start();  
      160.     }  
      161.   
      162.   
      163.     /** 
      164.      * 检查软件是否有更新版本 
      165.      * @return 
      166.      */  
      167.     public boolean isUpdate() {  
      168.         // 获取当前软件版本  
      169.         int versionCode = getVersionCode(mContext);  
      170.         //把version.xml放到网络上,然后获取文件信息  
      171.         InputStream inStream = ParseXmlService.class.getClassLoader().getResourceAsStream("version.xml");  
      172.         // 解析XML文件。 由于XML文件比较小,因此使用DOM方式进行解析  
      173.         ParseXmlService service = new ParseXmlService();  
      174.         try {  
      175.             mHashMap = service.parseXml(inStream);  
      176.         } catch (Exception e) {  
      177.             // TODO: handle exception  
      178.             e.printStackTrace();  
      179.         }  
      180.         if(null != mHashMap) {  
      181.             int serviceCode = Integer.valueOf(mHashMap.get("version"));  
      182.             //版本判断  
      183.             if(serviceCode > versionCode) {  
      184.                 return true;  
      185.             }  
      186.         }  
      187.         return false;  
      188.     }  
      189.   
      190.     /** 
      191.      * 获取软件版本号 
      192.      * @param context 
      193.      * @return 
      194.      */  
      195.     private int getVersionCode(Context context) {  
      196.         // TODO Auto-generated method stub  
      197.         int versionCode = 0;  
      198.   
      199.         // 获取软件版本号,对应AndroidManifest.xml下android:versionCode  
      200.         try {  
      201.             versionCode = context.getPackageManager().getPackageInfo(  
      202.                     "com.xiaowu.news", 0).versionCode;  
      203.         } catch (NameNotFoundException e) {  
      204.             // TODO Auto-generated catch block  
      205.             e.printStackTrace();  
      206.         }  
      207.         return versionCode;  
      208.     }  
      209.       
      210.     /** 
      211.      * 下载文件线程 
      212.      * @author Administrator 
      213.      * 
      214.      */  
      215.     private class DownloadApkThread extends Thread {  
      216.         @Override  
      217.         public void run() {  
      218.             try  
      219.             {  
      220.                 //判断SD卡是否存在,并且是否具有读写权限  
      221.                 if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED))  
      222.                 {  
      223.                     // 获取SDCard的路径  
      224.                     String sdpath = Environment.getExternalStorageDirectory() + "/";  
      225.                     mSavePath = sdpath + "download";  
      226.                     URL url = new URL(mHashMap.get("url"));  
      227.                     // 创建连接  
      228.                     HttpURLConnection conn = (HttpURLConnection) url.openConnection();  
      229.                     conn.connect();  
      230.                     // 获取文件大小  
      231.                     int length = conn.getContentLength();  
      232.                     // 创建输入流  
      233.                     InputStream is = conn.getInputStream();  
      234.   
      235.                     File file = new File(mSavePath);  
      236.                     // 如果文件不存在,新建目录  
      237.                     if (!file.exists())  
      238.                     {  
      239.                         file.mkdir();  
      240.                     }  
      241.                     File apkFile = new File(mSavePath, mHashMap.get("name"));  
      242.                     FileOutputStream fos = new FileOutputStream(apkFile);  
      243.                     int count = 0;  
      244.                     // 缓存  
      245.                     byte buf[] = new byte[1024];  
      246.                     // 写入到文件中  
      247.                     do  
      248.                     {  
      249.                         int numread = is.read(buf);  
      250.                         count += numread;  
      251.                         // 计算进度条的位置  
      252.                         progress = (int) (((float) count / length) * 100);  
      253.                         // 更新进度  
      254.                         mHandler.sendEmptyMessage(DOWNLOAD);  
      255.                         if (numread <= 0)  
      256.                         {  
      257.                             // 下载完成  
      258.                             mHandler.sendEmptyMessage(DOWNLOAD_FINISH);  
      259.                             break;  
      260.                         }  
      261.                         // 写入文件  
      262.                         fos.write(buf, 0, numread);  
      263.                     } while (!cancelUpdate);//点击取消就停止下载  
      264.                     fos.close();  
      265.                     is.close();  
      266.                 }  
      267.             } catch (MalformedURLException e)  
      268.             {  
      269.                 e.printStackTrace();  
      270.             } catch (IOException e)  
      271.             {  
      272.                 e.printStackTrace();  
      273.             }  
      274.             // 取消下载对话框显示  
      275.             mDownloadDialog.dismiss();  
      276.         }  
      277.     }  
      278.       
      279.     /** 
      280.      * 安装APK文件 
      281.      */  
      282.     private void installApk()  
      283.     {  
      284.         File apkfile = new File(mSavePath, mHashMap.get("name"));  
      285.         if (!apkfile.exists())  
      286.         {  
      287.             return;  
      288.         }  
      289.         Intent i = new Intent(Intent.ACTION_VIEW);  
      290.         i.setDataAndType(Uri.parse("file://" + apkfile.toString()), "application/vnd.android.package-archive");  
      291.         mContext.startActivity(i);  
      292.     }  
      293. }  
  • 相关阅读:
    推送消息为什么使用RocketMQ,而不使用Kafka?
    com.google.common.collect.Lists.addAll()空指针原因分析
    AQS原理
    ReentrantLock-加锁
    ReentrantLock-自旋
    Reentrantlock-的核心内容
    java中,BigDecimal的add方法避坑指南
    Reentrantlock-实现原理
    Reentrantlock-适用场景
    JAVA foreach和普通for循环是否需要判断为null
  • 原文地址:https://www.cnblogs.com/tc310/p/3962537.html
Copyright © 2020-2023  润新知