Android 轻松实现后台搭建+APP版本更新

简介:

转自:http://blog.csdn.net/u012422829/article/details/46355515

看完本文,您可以学到:


1.版本更新的方法

2.与后台的交互

3.Android中Handler的使用

4.Android中ProgressDialog的使用

话不多说,先来看看效果图:





一、大致思路阐述


首先,我们要有一个可以被手机访问的后台。
这里有 两种方法,在调试的时候我们可以利用 手机和笔记本连到同一个局域网的方式,在电脑上开启个类似PHP或者JAVAEE一样样的后台服务。
但是对于没有相关后台开发经验的朋友,这里有一种 更好的方式:利用Github等 免费空间来实现。详细请戳我的另一篇博文 利用Github建立你的个人网站 。
OK,有了存放资源的后台,我们要放点什么东西呢?很简单,一个 包含最新版本信息的update.txt文件和一个.apk文件足矣!

txt文件里写啥?看下我的例子:
XXX&1.3&这里写点描述&http://192.168.1.100:8080/PersonalHomePage/new.apk
解释一下:  &是分隔符,用于手机端获取到信息后的分割。 1.3代表着最新版本号,之后的是 新版本的描述,最后的是 新版本APK的下载地址(这里我用了局域网)。一开始的是啥呢?我当时在试验的时候,在开头并没有加额外信息,即以1.3开头,实验之后,发现手机端获取到TXT文本信息后不能正确解析,原因我觉得是因为TXT文件的开头包含有一些自带的字符,手机解析时会有问题。(感兴趣的朋友可以去深究,还望不吝赐教!)

OK,有了新版本的信息,我们要怎么做?
我们要获取到最新的版本号,然后与当前APP的版本号进行对比。如果低于最新版本,就到下载地址中去下载。

二、详细代码解释


首先,新建一个UpdateInfo类,用来与update.txt的内容对应,这个很简单:

[java]  view plain copy
  1. package com.example.appupdatedemo;  
  2.   
  3. public class UpdateInfo  
  4. {  
  5.         private String version;  
  6.         private String description;  
  7.         private String url;  
  8.           
  9.         public String getVersion()  
  10.         {  
  11.                 return version;  
  12.         }  
  13.         public void setVersion(String version)  
  14.         {  
  15.                 this.version = version;  
  16.         }  
  17.         public String getDescription()  
  18.         {  
  19.                 return description;  
  20.         }  
  21.         public void setDescription(String description)  
  22.         {  
  23.                 this.description = description;  
  24.         }  
  25.         public String getUrl()  
  26.         {  
  27.                 return url;  
  28.         }  
  29.         public void setUrl(String url)  
  30.         {  
  31.                 this.url = url;  
  32.         }  
  33.           
  34. }  


然后,写一个类去获取更新的信息,即我们的update.txt文件:

UpdateInfoService:

[java]  view plain copy
  1. package com.example.appupdatedemo;  
  2.   
  3. import java.io.BufferedReader;  
  4. import java.io.InputStreamReader;  
  5. import java.net.HttpURLConnection;  
  6. import java.net.URL;  
  7.   
  8. import android.content.Context;  
  9.   
  10. public class UpdateInfoService {  
  11.     public UpdateInfoService(Context context) {  
  12.     }  
  13.   
  14.     public UpdateInfo getUpDateInfo() throws Exception {  
  15.         String path = GetServerUrl.getUrl() + "/update.txt";  
  16.         StringBuffer sb = new StringBuffer();  
  17.         String line = null;  
  18.         BufferedReader reader = null;  
  19.         try {  
  20.             // 创建一个url对象  
  21.             URL url = new URL(path);  
  22.             // 通過url对象,创建一个HttpURLConnection对象(连接)  
  23.             HttpURLConnection urlConnection = (HttpURLConnection) url  
  24.                     .openConnection();  
  25.             // 通过HttpURLConnection对象,得到InputStream  
  26.             reader = new BufferedReader(new InputStreamReader(  
  27.                     urlConnection.getInputStream()));  
  28.             // 使用io流读取文件  
  29.             while ((line = reader.readLine()) != null) {  
  30.                 sb.append(line);  
  31.             }  
  32.         } catch (Exception e) {  
  33.             e.printStackTrace();  
  34.         } finally {  
  35.             try {  
  36.                 if (reader != null) {  
  37.                     reader.close();  
  38.                 }  
  39.             } catch (Exception e) {  
  40.                 e.printStackTrace();  
  41.             }  
  42.         }  
  43.         String info = sb.toString();  
  44.         UpdateInfo updateInfo = new UpdateInfo();  
  45.         updateInfo.setVersion(info.split("&")[1]);  
  46.         updateInfo.setDescription(info.split("&")[2]);  
  47.         updateInfo.setUrl(info.split("&")[3]);  
  48.         return updateInfo;  
  49.     }  
  50.   
  51. }  

这里获取文件的方法是先创建一个HttpURLConnection,再获取输入流。细心的朋友可能注意到其中有个类,叫GetServerUrl,这个类是用来存放后台地址信息的:

[java]  view plain copy
  1. package com.example.appupdatedemo;  
  2.   
  3. /** 
  4.  * 获取服务器IP地址 
  5.  */  
  6.   
  7. public class GetServerUrl{  
  8.     static String url="http://192.168.1.100:8080/PersonalHomePage";   //没错,我这里用的是本地的JAVAEE工程,各位根据实际情况修改。  
  9.               
  10.     public static String getUrl() {  
  11.         return url;  
  12.     }  
  13. }  

OK,到了这一步, 准备工作都做完了,接下来只剩一个类了!即我们的MainActicity,一共一百多行,我们分几部分来讲。

第一部分代码,做的工作是获取版本更新信息。

[java]  view plain copy
  1. public class MainActivity extends Activity {  
  2.   
  3.     // 更新版本要用到的一些信息  
  4.     private UpdateInfo info;  
  5.     private ProgressDialog pBar;  
  6.   
  7.     @Override  
  8.     protected void onCreate(Bundle savedInstanceState) {  
  9.         super.onCreate(savedInstanceState);  
  10.         setContentView(R.layout.activity_main);  
  11.   
  12.         Toast.makeText(MainActivity.this"正在检查版本更新..", Toast.LENGTH_SHORT).show();  
  13.         // 自动检查有没有新版本 如果有新版本就提示更新  
  14.         new Thread() {  
  15.             public void run() {  
  16.                 try {  
  17.                     UpdateInfoService updateInfoService = new UpdateInfoService(  
  18.                             MainActivity.this);  
  19.                     info = updateInfoService.getUpDateInfo();  
  20.                     handler1.sendEmptyMessage(0);  
  21.                 } catch (Exception e) {  
  22.                     e.printStackTrace();  
  23.                 }  
  24.             };  
  25.         }.start();  
  26.     }  
  27.   
  28.     @SuppressLint("HandlerLeak")  
  29.     private Handler handler1 = new Handler() {  
  30.         public void handleMessage(Message msg) {  
  31.             // 如果有更新就提示  
  32.             if (isNeedUpdate()) {   //在下面的代码段  
  33.                 showUpdateDialog();  //下面的代码段  
  34.             }  
  35.         };  
  36.     };  

这里我们用到了new Thread+ Handler的方式去进行异步加载版本信息,主要是因为在安卓中要把耗时任务放在非主线程中执行,否则会造成阻塞,抛出无响应异常。还有另外的实现方式是安卓封装的AsyncTask,具体可以参考这篇博文: Android AsyncTask详解


第二部分,判断是否是最新版本,如果不是,跳出对话框选择是否更新:

[java]  view plain copy
  1. private void showUpdateDialog() {  
  2.     AlertDialog.Builder builder = new AlertDialog.Builder(this);  
  3.     builder.setIcon(android.R.drawable.ic_dialog_info);  
  4.     builder.setTitle("请升级APP至版本" + info.getVersion());  
  5.     builder.setMessage(info.getDescription());  
  6.     builder.setCancelable(false);  
  7.   
  8.     builder.setPositiveButton("确定"new DialogInterface.OnClickListener() {  
  9.   
  10.         @Override  
  11.         public void onClick(DialogInterface dialog, int which) {  
  12.             if (Environment.getExternalStorageState().equals(  
  13.                     Environment.MEDIA_MOUNTED)) {  
  14.                 downFile(info.getUrl());     //在下面的代码段  
  15.             } else {  
  16.                 Toast.makeText(MainActivity.this"SD卡不可用,请插入SD卡",  
  17.                         Toast.LENGTH_SHORT).show();  
  18.             }  
  19.         }  
  20.     });  
  21.     builder.setNegativeButton("取消"new DialogInterface.OnClickListener() {  
  22.   
  23.         @Override  
  24.         public void onClick(DialogInterface dialog, int which) {  
  25.         }  
  26.   
  27.     });  
  28.     builder.create().show();  
  29. }  
  30.   
  31. private boolean isNeedUpdate() {  
  32.       
  33.     String v = info.getVersion(); // 最新版本的版本号  
  34.     Log.i("update",v);  
  35.     Toast.makeText(MainActivity.this, v, Toast.LENGTH_SHORT).show();  
  36.     if (v.equals(getVersion())) {  
  37.         return false;  
  38.     } else {  
  39.         return true;  
  40.     }  
  41. }  
  42.   
  43. // 获取当前版本的版本号  
  44. private String getVersion() {  
  45.     try {  
  46.         PackageManager packageManager = getPackageManager();  
  47.         PackageInfo packageInfo = packageManager.getPackageInfo(  
  48.                 getPackageName(), 0);  
  49.         return packageInfo.versionName;  
  50.     } catch (NameNotFoundException e) {  
  51.         e.printStackTrace();  
  52.         return "版本号未知";  
  53.     }  
  54. }  
这段里面要注意的是怎么获取当前版本,方法是使用PackageManager提供的getPackageInfo方法,返回的是manifest文件中的版本号。其他的代码挺简单,注释也挺全的。如果有问题,欢迎留言。

接下来是最后一部分,下载文件。

[java]  view plain copy
  1.     void downFile(final String url) {   
  2.         pBar = new ProgressDialog(MainActivity.this);    //进度条,在下载的时候实时更新进度,提高用户友好度  
  3.         pBar.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);  
  4.         pBar.setTitle("正在下载");  
  5.         pBar.setMessage("请稍候...");  
  6.         pBar.setProgress(0);  
  7.         pBar.show();  
  8.         new Thread() {  
  9.             public void run() {          
  10.                 HttpClient client = new DefaultHttpClient();  
  11.                 HttpGet get = new HttpGet(url);  
  12.                 HttpResponse response;  
  13.                 try {  
  14.                     response = client.execute(get);  
  15.                     HttpEntity entity = response.getEntity();  
  16.                     int length = (int) entity.getContentLength();   //获取文件大小  
  17.                                         pBar.setMax(length);                            //设置进度条的总长度  
  18.                     InputStream is = entity.getContent();  
  19.                     FileOutputStream fileOutputStream = null;  
  20.                     if (is != null) {  
  21.                         File file = new File(  
  22.                                 Environment.getExternalStorageDirectory(),  
  23.                                 "Test.apk");  
  24.                         fileOutputStream = new FileOutputStream(file);  
  25.                         byte[] buf = new byte[10];   //这个是缓冲区,即一次读取10个比特,我弄的小了点,因为在本地,所以数值太大一 下就下载完了,看不出progressbar的效果。  
  26.                         int ch = -1;  
  27.                         int process = 0;  
  28.                         while ((ch = is.read(buf)) != -1) {         
  29.                             fileOutputStream.write(buf, 0, ch);  
  30.                             process += ch;  
  31.                             pBar.setProgress(process);       //这里就是关键的实时更新进度了!  
  32.                         }  
  33.   
  34.                     }  
  35.                     fileOutputStream.flush();  
  36.                     if (fileOutputStream != null) {  
  37.                         fileOutputStream.close();  
  38.                     }  
  39.                     down();  
  40.                 } catch (ClientProtocolException e) {  
  41.                     e.printStackTrace();  
  42.                 } catch (IOException e) {  
  43.                     e.printStackTrace();  
  44.                 }  
  45.             }  
  46.   
  47.         }.start();  
  48.     }  
  49.   
  50.     void down() {  
  51.         handler1.post(new Runnable() {  
  52.             public void run() {  
  53.                 pBar.cancel();  
  54.                 update();  
  55.             }  
  56.         });  
  57.     }  
  58. //安装文件,一般固定写法  
  59.     void update() {                      
  60.         Intent intent = new Intent(Intent.ACTION_VIEW);  
  61.         intent.setDataAndType(Uri.fromFile(new File(Environment  
  62.                 .getExternalStorageDirectory(), "Test.apk")),  
  63.                 "application/vnd.android.package-archive");  
  64.         startActivity(intent);  
  65.     }  
这一段主要是利用progressdialog在下载的时候实时更新进度,主要利用的是一个字节数组的缓冲区。即每次获取到的内容填满缓冲区后就写入到本地本件中。这里我把缓冲区的大小设置为10个字节(1024会比较好),理由是因为在同一个局域网中速度特别快,刷一下就下载完了,看不出进度条效果,缓冲区调小点就OK了。

========================================

写在后面:

源代码已上传到我的Github,或者到CSDN下载区下载。

任何问题,欢迎留言交流!


相关文章
|
1月前
|
XML Java 数据库
安卓项目:app注册/登录界面设计
本文介绍了如何设计一个Android应用的注册/登录界面,包括布局文件的创建、登录和注册逻辑的实现,以及运行效果的展示。
151 0
安卓项目:app注册/登录界面设计
|
2月前
|
Java 数据库 Android开发
一个Android App最少有几个线程?实现多线程的方式有哪些?
本文介绍了Android多线程编程的重要性及其实现方法,涵盖了基本概念、常见线程类型(如主线程、工作线程)以及多种多线程实现方式(如`Thread`、`HandlerThread`、`Executors`、Kotlin协程等)。通过合理的多线程管理,可大幅提升应用性能和用户体验。
131 15
一个Android App最少有几个线程?实现多线程的方式有哪些?
|
2月前
|
存储 开发工具 Android开发
使用.NET MAUI开发第一个安卓APP
【9月更文挑战第24天】使用.NET MAUI开发首个安卓APP需完成以下步骤:首先,安装Visual Studio 2022并勾选“.NET Multi-platform App UI development”工作负载;接着,安装Android SDK。然后,创建新项目时选择“.NET Multi-platform App (MAUI)”模板,并仅针对Android平台进行配置。了解项目结构,包括`.csproj`配置文件、`Properties`配置文件夹、平台特定代码及共享代码等。
187 2
|
2月前
|
XML Android开发 数据格式
🌐Android国际化与本地化全攻略!让你的App走遍全球无障碍!🌍
在全球化背景下,实现Android应用的国际化与本地化至关重要。本文以一款旅游指南App为例,详细介绍如何通过资源文件拆分与命名、适配布局与方向、处理日期时间及货币格式、考虑文化习俗等步骤,完成多语言支持和本地化调整。通过邀请用户测试并收集反馈,确保应用能无缝融入不同市场,提升用户体验与满意度。
108 3
|
2月前
|
Java 数据库 Android开发
一个Android App最少有几个线程?实现多线程的方式有哪些?
本文介绍了Android应用开发中的多线程编程,涵盖基本概念、常见实现方式及最佳实践。主要内容包括主线程与工作线程的作用、多线程的多种实现方法(如 `Thread`、`HandlerThread`、`Executors` 和 Kotlin 协程),以及如何避免内存泄漏和合理使用线程池。通过有效的多线程管理,可以显著提升应用性能和用户体验。
74 10
|
1月前
|
安全 网络安全 Android开发
深度解析:利用Universal Links与Android App Links实现无缝网页至应用跳转的安全考量
【10月更文挑战第2天】在移动互联网时代,用户经常需要从网页无缝跳转到移动应用中。这种跳转不仅需要提供流畅的用户体验,还要确保安全性。本文将深入探讨如何利用Universal Links(仅限于iOS)和Android App Links技术实现这一目标,并分析其安全性。
246 0
|
2月前
|
XML 数据库 Android开发
10分钟手把手教你用Android手撸一个简易的个人记账App
该文章提供了使用Android Studio从零开始创建一个简单的个人记账应用的详细步骤,包括项目搭建、界面设计、数据库处理及各功能模块的实现方法。
|
JSON Java API
什么骚操作,用Android能写后台服务?
什么骚操作,用Android能写后台服务?
225 0
|
6天前
|
开发框架 前端开发 Android开发
安卓与iOS开发中的跨平台策略
在移动应用开发的战场上,安卓和iOS两大阵营各据一方。随着技术的演进,跨平台开发框架成为开发者的新宠,旨在实现一次编码、多平台部署的梦想。本文将探讨跨平台开发的优势与挑战,并分享实用的开发技巧,帮助开发者在安卓和iOS的世界中游刃有余。
|
11天前
|
搜索推荐 Android开发 开发者
探索安卓开发中的自定义视图:打造个性化UI组件
【10月更文挑战第39天】在安卓开发的世界中,自定义视图是实现独特界面设计的关键。本文将引导你理解自定义视图的概念、创建流程,以及如何通过它们增强应用的用户体验。我们将从基础出发,逐步深入,最终让你能够自信地设计和实现专属的UI组件。
下一篇
无影云桌面