本文转自:http://www.oschina.net/question/163910_28462
我们看到很多Android应用都具有自动更新功能,用户一键就可以完成软件的升级更新。得益于Android系统的软件包管理和安装机制,这一功能实现起来相当简单,下面我们就来实践一下。首先给出界面效果:

1. 准备知识 在AndroidManifest.xml里定义了每个Android apk的版本标识:
1 |
< manifest xmlns:android = "http://schemas.android.com/apk/res/android" |
3 |
android:versionCode = "1" |
4 |
android:versionName = "1.0.0" > |
5 |
< application ></ application > |
其中,android:versionCode和android:versionName两个字段分别表示版本代码,版本名称。versionCode是 整型数字,versionName是字符串。由于version是给用户看的,不太容易比较大小,升级检查时,可以以检查versionCode为主,方 便比较出版本的前后大小。 那么,在应用中如何读取AndroidManifest.xml中的versionCode和versionName呢?可以使用PackageManager的API,参考以下代码:
01 |
public static int getVerCode(Context context) { |
04 |
verCode = context.getPackageManager().getPackageInfo( |
05 |
"com.myapp" , 0 ).versionCode; |
06 |
} catch (NameNotFoundException e) { |
07 |
Log.e(TAG, e.getMessage()); |
12 |
public static String getVerName(Context context) { |
15 |
verName = context.getPackageManager().getPackageInfo( |
16 |
"com.myapp" , 0 ).versionName; |
17 |
} catch (NameNotFoundException e) { |
18 |
Log.e(TAG, e.getMessage()); |
或者在AndroidManifest中将android:versionName="1.2.0"写成 android:versionName="@string/app_versionName",然后在values/strings.xml中添加对应 字符串,这样实现之后,就可以使用如下代码获得版本名称:
1 |
public static String getVerName(Context context) { |
2 |
String verName = context.getResources() |
3 |
.getText(R.string.app_versionName).toString(); |
同理,apk的应用名称可以这样获得:
1 |
public static String getAppName(Context context) { |
2 |
String verName = context.getResources() |
3 |
.getText(R.string.app_name).toString(); |
2. 流程框架

3. 版本检查 在服务端放置最新版本的apk文件,如:http://localhost/myapp/myapp.apk 同时,在服务端放置对应此apk的版本信息调用接口或者文件,如:http://localhost/myapp/ver.json ver.json中的内容为:
[{"appname":"jtapp12","apkname":"jtapp-12-updateapksamples.apk","verName":1.0.1,"verCode":2}]
然后,在手机客户端上进行版本读取和检查:
01 |
private boolean getServerVer () { |
03 |
String verjson = NetworkTool.getContent(Config.UPDATE_SERVER |
04 |
+ Config.UPDATE_VERJSON); |
05 |
JSONArray array = new JSONArray(verjson); |
06 |
if (array.length() > 0 ) { |
07 |
JSONObject obj = array.getJSONObject( 0 ); |
09 |
newVerCode = Integer.parseInt(obj.getString( "verCode" )); |
10 |
newVerName = obj.getString( "verName" ); |
11 |
} catch (Exception e) { |
17 |
} catch (Exception e) { |
18 |
Log.e(TAG, e.getMessage()); |
比较服务器和客户端的版本,并进行更新操作。
1 |
if (getServerVerCode()) { |
2 |
int vercode = Config.getVerCode( this ); // 用到前面第一节写的方法 |
3 |
if (newVerCode > vercode) { |
4 |
doNewVersionUpdate(); // 更新新版本 |
6 |
notNewVersionShow(); // 提示当前为最新版本 |
详细方法:
01 |
private void notNewVersionShow() { |
02 |
int verCode = Config.getVerCode( this ); |
03 |
String verName = Config.getVerName( this ); |
04 |
StringBuffer sb = new StringBuffer(); |
09 |
sb.append( ",/n已是最新版,无需更新!" ); |
10 |
Dialog dialog = new AlertDialog.Builder(Update. this ).setTitle( "软件更新" ) |
11 |
.setMessage(sb.toString()) // 设置内容 |
12 |
.setPositiveButton( "确定" , // 设置确定按钮 |
13 |
new DialogInterface.OnClickListener() { |
15 |
public void onClick(DialogInterface dialog, |
23 |
private void doNewVersionUpdate() { |
24 |
int verCode = Config.getVerCode( this ); |
25 |
String verName = Config.getVerName( this ); |
26 |
StringBuffer sb = new StringBuffer(); |
31 |
sb.append( ", 发现新版本:" ); |
32 |
sb.append(newVerName); |
34 |
sb.append(newVerCode); |
36 |
Dialog dialog = new AlertDialog.Builder(Update. this ) |
38 |
.setMessage(sb.toString()) |
40 |
.setPositiveButton( "更新" , // 设置确定按钮 |
41 |
new DialogInterface.OnClickListener() { |
43 |
public void onClick(DialogInterface dialog, |
45 |
pBar = new ProgressDialog(Update. this ); |
46 |
pBar.setTitle( "正在下载" ); |
47 |
pBar.setMessage( "请稍候..." ); |
48 |
pBar.setProgressStyle(ProgressDialog.STYLE_SPINNER); |
49 |
downFile(Config.UPDATE_SERVER + Config.UPDATE_APKNAME); |
52 |
.setNegativeButton( "暂不更新" , |
53 |
new DialogInterface.OnClickListener() { |
54 |
public void onClick(DialogInterface dialog, |
4. 下载模块
注,本部分参考了前人的相关实现,见 http://apps.hi.baidu.com/share/detail/24172508
01 |
void downFile( final String url) { |
05 |
HttpClient client = new DefaultHttpClient(); |
06 |
HttpGet get = new HttpGet(url); |
07 |
HttpResponse response; |
09 |
response = client.execute(get); |
10 |
HttpEntity entity = response.getEntity(); |
11 |
long length = entity.getContentLength(); |
12 |
InputStream is = entity.getContent(); |
13 |
FileOutputStream fileOutputStream = null ; |
16 |
Environment.getExternalStorageDirectory(), |
17 |
Config.UPDATE_SAVENAME); |
18 |
fileOutputStream = new FileOutputStream(file); |
19 |
byte [] buf = new byte [ 1024 ]; |
22 |
while ((ch = is.read(buf)) != - 1 ) { |
23 |
fileOutputStream.write(buf, 0 , ch); |
29 |
fileOutputStream.flush(); |
30 |
if (fileOutputStream != null ) { |
31 |
fileOutputStream.close(); |
34 |
} catch (ClientProtocolException e) { |
36 |
} catch (IOException e) { |
下载完成,通过handler通知主ui线程将下载对话框取消。
2 |
handler.post( new Runnable() { |
5. 安装应用
1 |
<strong> void update() { |
2 |
Intent intent = new Intent(Intent.ACTION_VIEW); |
3 |
intent.setDataAndType(Uri.fromFile( new File(Environment |
4 |
.getExternalStorageDirectory(), Config.UPDATE_SAVENAME)), |
5 |
"application/vnd.android.package-archive" ); |
如果你将apk应用发布到market上,那么,你会发现market内建了类似的模块,可以自动更新或者提醒你是否更新应用。那么,对于你自己的 应用需要自动更新的话,自己内建一个是不是更加方便了呢?本文提到的代码大多是在UpdateActivity.java中实现,为了能够使更新过程更加 友好,可以在最初launcher的Activity中建立一个线程,用来检查服务端是否有更新。有更新的时候就启动UpdateActivity,这样 的使用体验更加平滑。
本文例程源码查看/下载: http://code.google.com/p/androidex/source/browse/trunk/jtapp-12-updateapksamples
版权归个人所有,转载请注明出处
http://blog.csdn.net/xjanker2/archive/2011/04/06/6303937.aspx
- IoTClientTool自动升级更新
IoTClientTool是什么 IoTClientTool是什么,IoTClientTool是IoTClient开源组件的可视化操的作实现.方便对plc设备和ModBusRtu.BACnet.串口等 ...
- Android应用的自动升级、更新模块的实现(转)
我们看到很多Android应用都具有自动更新功能,用户一键就可以完成软件的升级更新.得益于Android系统的软件包管理和安装机制,这一功能实现起来相当简单,下面我们就来实践一下.首先给出界面效果: ...
- Android应用的自动升级、更新模块的实现
我们看到很多Android应用都具有自动更新功能,用户一键就可以完成软件的升级更新.得益于Android系统的软件包管理和安装机制,这一功能实现起来相当简单,下面我们就来实践一下.首先给出界面效果: ...
- Winform开发框架之通用自动更新模块(转)
在网络化的环境中,特别是基于互联网发布的Winform程序,程序的自动更新功能是比较重要的操作,这样可以避免挨个给使用者打电话.发信息通知或者发送软件等,要求其对应用程序进行升级.实现程序的自动更新, ...
- C# Xamarin For Android自动升级项目实战
一.课程介绍 “明人不说暗话,跟着阿笨一起玩Xamarin”,本次分享课程阿笨将带来大家一起学习Xamarin For Android系列<C# Xamarin For Android自动升级项 ...
- Ionic实战 自动升级APP(Android版)
Ionic 框架介绍 Ionic是一个基于Angularjs.可以使用HTML5构建混合移动应用的用户界面框架,它自称为是"本地与HTML5的结合".该框架提供了很多基本的移动用户 ...
- Android 实现应用升级方案(暨第三方自动升级服务无法使用后的解决方案)
第三方推送升级服务不再靠谱: 以前在做Android开发的时候,在应用升级方面都是使用的第三方推送升级服务,但是目前因为一些非技术性的问题,一些第三方厂商不再提供自动升级服务,比如友盟,那么当第三方推 ...
- Android自动检测版本及自动升级
步骤: 1.检测当前版本的信息AndroidManifest.xml-->manifest-->android:versionName. 2.从服务器获取版本号(版本号存在于xml文件中) ...
- Android 一s个相对完整的自动升级功能实现代码
由于项目的需要最近做了一个关于Android自动升级的功能,下面将贴出Android手机客户端的完整代码.这段代码参考别的代码居多,由于不满足需求,所以自己仅仅改了一些需要变动的内容,其他功能都是按照 ...
随机推荐
- [Sparrow OS 设计文档连载(一)] Introduction
- MFC容器类介绍
我们知道如果是单个的少数几个值弄些int , long,float ,double等类型的变量来装这些值就行了.但如果值太多这样就比较麻烦.当然数据超级多时就直接放数据库里存着去了. 但如果数值不多不 ...
- 快速找到跟踪其他session产生的trc文件
掌握该技术最根本的是需要搞清楚session跟踪文件存放的路径和生成跟踪文件的命名规则,不然,在已经存在成全上万trc文件的生产环境中,要想快速正确的找到跟踪其他SESSION产生的trc文件就如大海 ...
- udt nat traverse
https://github.com/bakwc/udt-nat-traverse Example of nat traversal using udt library. UDT is a udp b ...
- 【M9】利用destructors避免泄漏资源
1.在堆上获取的动态资源,用户忘记delete,或者由于异常导致没有没执行到delete,都会造成资源泄漏. 2.我们知道,栈上的对象,离开作用域,必定要执行析构方法.即使抛出异常,会堆栈回滚,保证已 ...
- Codeforces Educational Codeforces Round 3 E. Minimum spanning tree for each edge LCA链上最大值
E. Minimum spanning tree for each edge 题目连接: http://www.codeforces.com/contest/609/problem/E Descrip ...
- Java IO之File
FILE类是用来实现获取文件.文件夹的类库工具,File并不是像类名所表示的那样仅仅是用来表示文件.它还能够用来表示文件夹. 所以能够用File来获取一个文件夹下的全部文件,甚至是文件夹中的文件. 一 ...
- android复制数据库到SD卡(网上搜集,未经验证)
android中使用sqlite.复制assets下的数据库到SD卡.支持大于1M的文件 如果使用SD卡,需要在AndroidManifest.xml中设置权限 <uses-permission ...
- STL中的优先级队列(priority_queue)的自己实现priqueue
这篇文章主要介绍堆(最大堆和最小堆),以及一些系统对一些任务,比如线程,进程做调度的时候,所采用的优先级队列. 主要思想就是,做一个最大堆(任务的权重最大的在顶端),把顶端的任务取出,重新做一个堆,处 ...
- iOS开发——数据持久化Swift篇&(二)沙盒文件
沙盒文件 //******************** 5.2 文件操作 func use_FileOperations() { //1.获取程序的Home目录 let homeDirectory = ...