- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章Android编程实现应用自动更新、下载、安装的方法由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
本文实例讲述了android编程实现应用自动更新、下载、安装的方法。分享给大家供大家参考,具体如下:
我们看到很多android应用都具有自动更新功能,用户一键就可以完成软件的升级更新。得益于android系统的软件包管理和安装机制,这一功能实现起来相当简单,下面我们就来实践一下.
1. 准备知识 。
在androidmanifest.xml里定义了每个android apk的版本标识:
1
2
3
4
5
6
|
<manifest xmlns:android=
"http://schemas.android.com/apk/res/android"
package
=
"com.myapp"
android:versioncode=
"1"
android:versionname=
"1.0.0"
>
<application></application>
</manifest>
|
其中,android:versioncode和android:versionname两个字段分别表示版本代码,版本名称。versioncode是 整型数字,versionname是字符串。由于version是给用户看的,不太容易比较大小,升级检查时,可以以检查versioncode为主,方 便比较出版本的前后大小.
那么,在应用中如何读取androidmanifest.xml中的versioncode和versionname呢?可以使用packagemanager的api,参考以下代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
public
static
int
getvercode(context context) {
int
vercode = -
1
;
try
{
vercode = context.getpackagemanager().getpackageinfo(
"com.myapp"
,
0
).versioncode;
}
catch
(namenotfoundexception e) {
log.e(tag, e.getmessage());
}
return
vercode;
}
public
static
string getvername(context context) {
string vername =
""
;
try
{
vername = context.getpackagemanager().getpackageinfo(
"com.myapp"
,
0
).versionname;
}
catch
(namenotfoundexception e) {
log.e(tag, e.getmessage());
}
return
vername;
}
|
或者在androidmanifest中将android:versionname=”1.2.0″写成 android:versionname=”@string/app_versionname”,然后在values/strings.xml中添加对应 字符串,这样实现之后,就可以使用如下代码获得版本名称:
1
2
3
4
5
|
public
static
string getvername(context context) {
string vername = context.getresources()
.gettext(r.string.app_versionname).tostring();
return
vername;
}
|
同理,apk的应用名称可以这样获得:
1
2
3
4
5
|
public
static
string getappname(context context) {
string vername = context.getresources()
.gettext(r.string.app_name).tostring();
return
vername;
}
|
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}] 。
然后,在手机客户端上进行版本读取和检查:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
private
boolean
getserverver () {
try
{
string verjson = networktool.getcontent(config.update_server
+ config.update_verjson);
jsonarray array =
new
jsonarray(verjson);
if
(array.length() >
0
) {
jsonobject obj = array.getjsonobject(
0
);
try
{
newvercode = integer.parseint(obj.getstring(
"vercode"
));
newvername = obj.getstring(
"vername"
);
}
catch
(exception e) {
newvercode = -
1
;
newvername =
""
;
return
false
;
}
}
}
catch
(exception e) {
log.e(tag, e.getmessage());
return
false
;
}
return
true
;
}
|
比较服务器和客户端的版本,并进行更新操作.
1
2
3
4
5
6
7
8
|
if
(getserververcode()) {
int
vercode = config.getvercode(
this
);
// 用到前面第一节写的方法
if
(newvercode > vercode) {
donewversionupdate();
// 更新新版本
}
else
{
notnewversionshow();
// 提示当前为最新版本
}
}
|
详细方法:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
|
private
void
notnewversionshow() {
int
vercode = config.getvercode(
this
);
string vername = config.getvername(
this
);
stringbuffer sb =
new
stringbuffer();
sb.append(
"当前版本:"
);
sb.append(vername);
sb.append(
" code:"
);
sb.append(vercode);
sb.append(
",/n已是最新版,无需更新!"
);
dialog dialog =
new
alertdialog.builder(update.
this
).settitle(
"软件更新"
)
.setmessage(sb.tostring())
// 设置内容
.setpositivebutton(
"确定"
,
// 设置确定按钮
new
dialoginterface.onclicklistener() {
@override
public
void
onclick(dialoginterface dialog,
int
which) {
finish();
}
}).create();
// 创建
// 显示对话框
dialog.show();
}
private
void
donewversionupdate() {
int
vercode = config.getvercode(
this
);
string vername = config.getvername(
this
);
stringbuffer sb =
new
stringbuffer();
sb.append(
"当前版本:"
);
sb.append(vername);
sb.append(
" code:"
);
sb.append(vercode);
sb.append(
", 发现新版本:"
);
sb.append(newvername);
sb.append(
" code:"
);
sb.append(newvercode);
sb.append(
", 是否更新?"
);
dialog dialog =
new
alertdialog.builder(update.
this
)
.settitle(
"软件更新"
)
.setmessage(sb.tostring())
// 设置内容
.setpositivebutton(
"更新"
,
// 设置确定按钮
new
dialoginterface.onclicklistener() {
@override
public
void
onclick(dialoginterface dialog,
int
which) {
pbar =
new
progressdialog(update.
this
);
pbar.settitle(
"正在下载"
);
pbar.setmessage(
"请稍候..."
);
pbar.setprogressstyle(progressdialog.style_spinner);
downfile(config.update_server + config.update_apkname);
}
})
.setnegativebutton(
"暂不更新"
,
new
dialoginterface.onclicklistener() {
public
void
onclick(dialoginterface dialog,
int
whichbutton) {
// 点击"取消"按钮之后退出程序
finish();
}
}).create();
// 创建
// 显示对话框
dialog.show();
}
|
4. 下载模块 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
|
void
downfile(
final
string url) {
pbar.show();
new
thread() {
public
void
run() {
httpclient client =
new
defaulthttpclient();
httpget get =
new
httpget(url);
httpresponse response;
try
{
response = client.execute(get);
httpentity entity = response.getentity();
long
length = entity.getcontentlength();
inputstream is = entity.getcontent();
fileoutputstream fileoutputstream =
null
;
if
(is !=
null
) {
file file =
new
file(
environment.getexternalstoragedirectory(),
config.update_savename);
fileoutputstream =
new
fileoutputstream(file);
byte
[] buf =
new
byte
[
1024
];
int
ch = -
1
;
int
count =
0
;
while
((ch = is.read(buf)) != -
1
) {
fileoutputstream.write(buf,
0
, ch);
count += ch;
if
(length >
0
) {
}
}
}
fileoutputstream.flush();
if
(fileoutputstream !=
null
) {
fileoutputstream.close();
}
down();
}
catch
(clientprotocolexception e) {
e.printstacktrace();
}
catch
(ioexception e) {
e.printstacktrace();
}
}
}.start();
}
|
下载完成,通过handler通知主ui线程将下载对话框取消.
1
2
3
4
5
6
7
8
|
void
down() {
handler.post(
new
runnable() {
public
void
run() {
pbar.cancel();
update();
}
});
}
|
5. 安装应用 。
1
2
3
4
5
6
7
|
void
update() {
intent intent =
new
intent(intent.action_view);
intent.setdataandtype(uri.fromfile(
new
file(environment
.getexternalstoragedirectory(), config.update_savename)),
"application/vnd.android.package-archive"
);
startactivity(intent);
}
|
如果你将apk应用发布到market上,那么,你会发现market内建了类似的模块,可以自动更新或者提醒你是否更新应用。那么,对于你自己的 应用需要自动更新的话,自己内建一个是不是更加方便了呢?本文提到的代码大多是在updateactivity.java中实现,为了能够使更新过程更加 友好,可以在最初launcher的activity中建立一个线程,用来检查服务端是否有更新。有更新的时候就启动updateactivity,这样 的使用体验更加平滑.
希望本文所述对大家android程序设计有所帮助.
最后此篇关于Android编程实现应用自动更新、下载、安装的方法的文章就讲到这里了,如果你想了解更多关于Android编程实现应用自动更新、下载、安装的方法的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我查看了网站上的一些问题,但还没有完全弄清楚我做错了什么。我有一些这样的代码: var mongoose = require('mongoose'), db = mongoose.connect('m
基本上,根据 this bl.ocks,我试图在开始新序列之前让所有 block 都变为 0。我认为我需要的是以下顺序: 更新为0 退出到0 更新随机数 输入新号码 我尝试通过添加以下代码块来遵循上述
我试图通过使用随机数在循环中设置 JSlider 位置来模拟“赛马”的投注结果。我的问题是,当然,我无法在线程执行时更新 GUI,因此我的 JSlider 似乎没有在竞赛,它们从头到尾都在运行。我尝试
该功能非常简单: 变量:$table是正在更新的表$fields 是表中的字段,$values 从帖子生成并放入 $values 数组中而$where是表的索引字段的id值$indxfldnm 是索引
让我们想象一个环境:有一个数据库客户端和一个数据库服务器。数据库客户端可以是 Java 程序或其他程序等;数据库服务器可以是mysql、oracle等。 需求是在数据库服务器上的一个表中插入大量记录。
在我当前的应用程序中,我正在制作一个菜单结构,它可以递归地创建自己的子菜单。然而,由于这个原因,我发现很难也允许某种重新排序方法。大多数应用程序可能只是通过“排序”列进行排序,但是在这种情况下,尽管这
Provisioning Profile 有 key , key 链依赖于它。我想知道 key 什么时候会改变。 Key will change after renew Provisioning Pr
截至目前,我在\server\publications.js 中有我的 MongoDB“选择”,例如: Meteor.publish("jobLocations", function () { r
我读到 UI 应该始终在主线程上更新。但是,当谈到实现这些更新的首选方法时,我有点困惑。 我有各种函数可以执行一些条件检查,然后使用结果来确定如何更新 UI。我的问题是整个函数应该在主线程上运行吗?应
我在代理后面,我无法构建 Docker 镜像。 我试过 FROM ubuntu , FROM centos和 FROM alpine ,但是 apt-get update/yum update/apk
我构建了一个 Java 应用程序,它向外部授权客户端公开网络服务。 Web 服务使用带有证书身份验证的 WS-security。基本上我们充当自定义证书颁发机构 - 我们在我们的服务器上维护一个 ja
因此,我有时会在上传新版本时使用 app_offline.htm 使应用程序离线。 但是,当我上传较大的 dll 时,我收到黄色错误屏幕,指出无法加载 dll。 这似乎与我对 app_offline.
我刚刚下载了 VS Apache Cordova Tools Update 5,但遇到了 Node 和 NPM 的问题。我使用默认的空白 cordova 项目进行测试。 版本 如果我在 VS 项目中对
所以我有一个使用传单库实例化的 map 对象。 map 实例在单独的模板中创建并以这种方式路由:- var app = angular.module('myApp', ['ui', 'ngResour
我使用较早的 Java 6 u 3 获得的帧速率是新版本的两倍。很奇怪。谁能解释一下? 在 Core 2 Duo 1.83ghz 上,集成视频(仅使用一个内核)- 1500(较旧的 java)与 70
我正在使用 angular 1.2 ng-repeat 创建的 div 也包含 ng-click 点击时 ng-click 更新 $scope $scope 中的变化反射(reflect)在使用 $a
这些方法有什么区别 public final void moveCamera(CameraUpdate更新)和public final void animateCamera (CameraUpdate
我尝试了另一篇文章中某人评论中关于如何将树更改为列表的建议。但是,我在某处(或某物)有未声明的变量,所以我列表中的值是 [_G667, _G673, _G679],而不是 [5, 2, 6],这是正确
实现以下场景的最佳方法是什么? 我需要从java应用程序调用/查询包含数百万条记录的数据库表。然后,对于表中的每条记录,我的应用程序应该调用第三方 API 并获取状态字段作为响应。然后我的应用程序应该
只是在编写一些与 java 图形相关的代码,这是我今天的讲座中的非常简单的示例。不管怎样,互联网似乎说更新不会被系统触发器调用,例如调整框架大小等。在这个例子中,更新是由这样的触发器调用的(因此当我只
我是一名优秀的程序员,十分优秀!