在应用中, 为了提高用户体验, 会提供更新版本的功能. 那么如何实现呢? 我写了一个简单的Demo, 说明一下, 需要注意几个细节. 使用了Retrofit和Rx处理网络请求.
Github下载地址
1. 逻辑
访问服务器, 根据是否包含新版本, 判断是否需要更新.
下载Apk, 下载完成后, 自动安装, 高版本会覆盖低版本.
逻辑:
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
|
public class MainActivity extends AppCompatActivity {
private static final String APP_NAME = "Ped_android" ;
private static final String VERSION = "1.0.0" ;
private static final String INFO_NAME = "计步器" ;
private static final String STORE_APK = "chunyu_apk" ;
@Bind (R.id.main_b_install_apk) Button mBInstallApk;
private UpdateAppUtils.UpdateCallback mUpdateCallback; // 更新回调
@Override
protected void onCreate(Bundle savedInstanceState) {
super .onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ButterKnife.bind( this );
mUpdateCallback = new UpdateAppUtils.UpdateCallback() {
@Override public void onSuccess(UpdateInfo updateInfo) {
Toast.makeText(MainActivity. this , "有更新" , Toast.LENGTH_SHORT).show();
UpdateAppUtils.downloadApk(MainActivity. this , updateInfo, INFO_NAME, STORE_APK);
}
@Override public void onError() {
Toast.makeText(MainActivity. this , "无更新" , Toast.LENGTH_SHORT).show();
}
};
mBInstallApk.setOnClickListener( new View.OnClickListener() {
@Override public void onClick(View v) {
UpdateAppUtils.checkUpdate(APP_NAME, VERSION, mUpdateCallback);
}
});
}
} |
UpdateAppUtils
是核心下载类. 输入App的代号, 版本号, 异步回调, 发送到服务器, 判断是否需要更新. 如果存在新版本, 则下载Apk, 并自动安装更新.
2. 网络请求
更新请求, 参数是App代号和当前版本号.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
/** * 更新服务
* <p>
* Created by wangchenlong on 16/1/4.
*/
public interface UpdateService {
// 获取个人信息
@GET ( "/cmsapi/app/update" )
Observable<UpdateInfo> getUpdateInfo(
@Query ( "appName" ) String appName,
@Query ( "version" ) String version);
} |
创建服务的工厂类.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
/** * 创建Retrofit服务
* <p>
* Created by wangchenlong on 16/1/4.
*/
public class ServiceFactory {
public static <T> T createServiceFrom( final Class<T> serviceClass, String endpoint) {
Retrofit adapter = new Retrofit.Builder()
.baseUrl(endpoint)
.addCallAdapterFactory(RxJavaCallAdapterFactory.create()) // 添加Rx适配器
.addConverterFactory(GsonConverterFactory.create()) // 添加Gson转换器
.build();
return adapter.create(serviceClass);
}
} |
更新信息的Json类.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
/** * 更新信息(JSON)
* <p>
* Created by wangchenlong on 16/1/4.
*/
public class UpdateInfo {
public Data data; // 信息
public Integer error_code; // 错误代码
public String error_msg; // 错误信息
public static class Data {
public String curVersion; // 当前版本
public String appURL; // 下载地址
public String description; // 描述
public String minVersion; // 最低版本
public String appName; // 应用名称
}
@Override public String toString() {
return "当前版本: " + data.curVersion + ", 下载地址: " + data.appURL + ", 描述信息: " + data.description
+ ", 最低版本: " + data.minVersion + ", 应用代称: " + data.appName
+ ", 错误代码: " + error_code + ", 错误信息: " + error_msg;
}
} |
3. 请求和下载
更新库的主类, 包含检查更新(checkUpdate)
和下载Apk(downloadApk)
两个重要方法.
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
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
|
/** * 更新管理器
* <p>
* Created by wangchenlong on 16/1/6.
*/
@SuppressWarnings ( "unused" )
public class UpdateAppUtils {
@SuppressWarnings ( "unused" )
private static final String TAG = "DEBUG-WCL: " + UpdateAppUtils. class .getSimpleName();
/**
* 检查更新
*/
@SuppressWarnings ( "unused" )
public static void checkUpdate(String appCode, String curVersion, UpdateCallback updateCallback) {
UpdateService updateService =
ServiceFactory.createServiceFrom(UpdateService. class , UpdateService.ENDPOINT);
updateService.getUpdateInfo(appCode, curVersion)
.subscribeOn(Schedulers.newThread())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(updateInfo -> onNext(updateInfo, updateCallback),
throwable -> onError(throwable, updateCallback));
}
// 显示信息
private static void onNext(UpdateInfo updateInfo, UpdateCallback updateCallback) {
Log.e(TAG, "返回数据: " + updateInfo.toString());
if (updateInfo.error_code != 0 || updateInfo.data == null ||
updateInfo.data.appURL == null ) {
updateCallback.onError(); // 失败
} else {
updateCallback.onSuccess(updateInfo);
}
}
// 错误信息
private static void onError(Throwable throwable, UpdateCallback updateCallback) {
updateCallback.onError();
}
/**
* 下载Apk, 并设置Apk地址,
* 默认位置: /storage/sdcard0/Download
*
* @param context 上下文
* @param updateInfo 更新信息
* @param infoName 通知名称
* @param storeApk 存储的Apk
*/
@SuppressWarnings ( "unused" )
public static void downloadApk(
Context context, UpdateInfo updateInfo,
String infoName, String storeApk
) {
if (!isDownloadManagerAvailable()) {
return ;
}
String description = updateInfo.data.description;
String appUrl = updateInfo.data.appURL;
if (appUrl == null || appUrl.isEmpty()) {
Log.e(TAG, "请填写\"App下载地址\"" );
return ;
}
appUrl = appUrl.trim(); // 去掉首尾空格
if (!appUrl.startsWith( "http" )) {
}
Log.e(TAG, "appUrl: " + appUrl);
DownloadManager.Request request;
try {
request = new DownloadManager.Request(Uri.parse(appUrl));
} catch (Exception e) {
e.printStackTrace();
return ;
}
request.setTitle(infoName);
request.setDescription(description);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
request.allowScanningByMediaScanner();
request.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED);
}
request.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, storeApk);
Context appContext = context.getApplicationContext();
DownloadManager manager = (DownloadManager)
appContext.getSystemService(Context.DOWNLOAD_SERVICE);
// 存储下载Key
SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(appContext);
sp.edit().putLong(PrefsConsts.DOWNLOAD_APK_ID_PREFS, manager.enqueue(request)).apply();
}
// 最小版本号大于9
private static boolean isDownloadManagerAvailable() {
return Build.VERSION.SDK_INT >= Build.VERSION_CODES.GINGERBREAD;
}
// 错误回调
public interface UpdateCallback {
void onSuccess(UpdateInfo updateInfo);
void onError();
}
} |
检查更新: 创建服务, 在新线程中发送请求, 在主线程中接收数据, 判断成功和失败.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
/** * 检查更新
*/
@SuppressWarnings ( "unused" )
public static void checkUpdate(String appCode, String curVersion, UpdateCallback updateCallback) {
UpdateService updateService =
ServiceFactory.createServiceFrom(UpdateService. class , UpdateService.ENDPOINT);
updateService.getUpdateInfo(appCode, curVersion)
.subscribeOn(Schedulers.newThread())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(updateInfo -> onNext(updateInfo, updateCallback),
throwable -> onError(throwable, updateCallback));
}
|
下载Apk: 转换和解析Url, 设置通知信息和存储位置, 存储下载Id, 自动安装更新.
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
|
/** * 下载Apk, 并设置Apk地址,
* 默认位置: /storage/sdcard0/Download
*
* @param context 上下文
* @param updateInfo 更新信息
* @param infoName 通知名称
* @param storeApk 存储的Apk
*/
@SuppressWarnings ( "unused" )
public static void downloadApk(
Context context, UpdateInfo updateInfo,
String infoName, String storeApk
) { if (!isDownloadManagerAvailable()) {
return ;
}
String description = updateInfo.data.description;
String appUrl = updateInfo.data.appURL;
if (appUrl == null || appUrl.isEmpty()) {
Log.e(TAG, "请填写\"App下载地址\"" );
return ;
}
appUrl = appUrl.trim(); // 去掉首尾空格
if (!appUrl.startsWith( "http" )) {
}
Log.e(TAG, "appUrl: " + appUrl);
DownloadManager.Request request;
try {
request = new DownloadManager.Request(Uri.parse(appUrl));
} catch (Exception e) {
e.printStackTrace();
return ;
}
request.setTitle(infoName);
request.setDescription(description);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
request.allowScanningByMediaScanner();
request.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED);
}
request.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, storeApk);
Context appContext = context.getApplicationContext();
DownloadManager manager = (DownloadManager)
appContext.getSystemService(Context.DOWNLOAD_SERVICE);
// 存储下载Key
SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(appContext);
sp.edit().putLong(PrefsConsts.DOWNLOAD_APK_ID_PREFS, manager.enqueue(request)).apply();
} |
使用
DownloadManager
下载文件是Android的推荐方式.
存储下载Id(manager.enqueue(request))
是为了在安装应用时, 找到Apk.
默认存储地址/storage/sdcard0/Download
.
4.自动安装
注册广播接收器, 接收消息ACTION_DOWNLOAD_COMPLETE
, 下载完成会发送广播. 获取下载文件的Uri, 进行匹配, 发送安装消息, 自动安装.
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
|
/** * 安装下载接收器
* <p>
* Created by wangchenlong on 16/1/5.
*/
public class InstallReceiver extends BroadcastReceiver {
private static final String TAG =
"DEBUG-WCL: " + InstallReceiver. class .getSimpleName();
// 安装下载接收器
@Override public void onReceive(Context context, Intent intent) {
if (intent.getAction().equals(DownloadManager.ACTION_DOWNLOAD_COMPLETE)) {
long downloadApkId = intent.getLongExtra(DownloadManager.EXTRA_DOWNLOAD_ID, - 1 );
installApk(context, downloadApkId);
}
}
// 安装Apk
private void installApk(Context context, long downloadApkId) {
// 获取存储ID
SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(context);
long id = sp.getLong(PrefsConsts.DOWNLOAD_APK_ID_PREFS, -1L);
if (downloadApkId == id) {
DownloadManager dManager = (DownloadManager) context.getSystemService(Context.DOWNLOAD_SERVICE);
Intent install = new Intent(Intent.ACTION_VIEW);
Uri downloadFileUri = dManager.getUriForDownloadedFile(downloadApkId);
if (downloadFileUri != null ) {
install.setDataAndType(downloadFileUri, "application/vnd.android.package-archive" );
install.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(install);
} else {
Log.e(TAG, "下载失败" );
}
}
}
} |
安装本应用下载的Apk, 不安装其他Apk, 存储下载Id, 与广播Id进行匹配.
下载失败, 也会发送下载完成(ACTION_DOWNLOAD_COMPLETE)
广播, Uri可能为空, 需要判断, 否则发生崩溃.在应用中, 为了提高用户体验, 会提供更新版本的功能. 那么如何实现呢? 我写了一个简单的Demo, 说明一下, 需要注意几个细节. 使用了Retrofit和Rx处理网络请求.
Github下载地址
1. 逻辑
访问服务器, 根据是否包含新版本, 判断是否需要更新.
下载Apk, 下载完成后, 自动安装, 高版本会覆盖低版本.
逻辑:
1234567891011121314151617181920212223242526272829303132333435public
class
MainActivity
extends
AppCompatActivity {
private
static
final
String APP_NAME =
"Ped_android"
;
private
static
final
String VERSION =
"1.0.0"
;
private
static
final
String INFO_NAME =
"计步器"
;
private
static
final
String STORE_APK =
"chunyu_apk"
;
@Bind
(R.id.main_b_install_apk) Button mBInstallApk;
private
UpdateAppUtils.UpdateCallback mUpdateCallback;
// 更新回调
@Override
protected
void
onCreate(Bundle savedInstanceState) {
super
.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ButterKnife.bind(
this
);
mUpdateCallback =
new
UpdateAppUtils.UpdateCallback() {
@Override
public
void
onSuccess(UpdateInfo updateInfo) {
Toast.makeText(MainActivity.
this
,
"有更新"
, Toast.LENGTH_SHORT).show();
UpdateAppUtils.downloadApk(MainActivity.
this
, updateInfo, INFO_NAME, STORE_APK);
}
@Override
public
void
onError() {
Toast.makeText(MainActivity.
this
,
"无更新"
, Toast.LENGTH_SHORT).show();
}
};
mBInstallApk.setOnClickListener(
new
View.OnClickListener() {
@Override
public
void
onClick(View v) {
UpdateAppUtils.checkUpdate(APP_NAME, VERSION, mUpdateCallback);
}
});
}
}
UpdateAppUtils
是核心下载类. 输入App的代号, 版本号, 异步回调, 发送到服务器, 判断是否需要更新. 如果存在新版本, 则下载Apk, 并自动安装更新.2. 网络请求
更新请求, 参数是App代号和当前版本号.
1234567891011121314/**
* 更新服务
* <p>
* Created by wangchenlong on 16/1/4.
*/
public
interface
UpdateService {
// 获取个人信息
@GET
(
"/cmsapi/app/update"
)
Observable<UpdateInfo> getUpdateInfo(
@Query
(
"appName"
) String appName,
@Query
(
"version"
) String version);
}
创建服务的工厂类.
123456789101112131415/**
* 创建Retrofit服务
* <p>
* Created by wangchenlong on 16/1/4.
*/
public
class
ServiceFactory {
public
static
<T> T createServiceFrom(
final
Class<T> serviceClass, String endpoint) {
Retrofit adapter =
new
Retrofit.Builder()
.baseUrl(endpoint)
.addCallAdapterFactory(RxJavaCallAdapterFactory.create())
// 添加Rx适配器
.addConverterFactory(GsonConverterFactory.create())
// 添加Gson转换器
.build();
return
adapter.create(serviceClass);
}
}
更新信息的Json类.
123456789101112131415161718192021222324/**
* 更新信息(JSON)
* <p>
* Created by wangchenlong on 16/1/4.
*/
public
class
UpdateInfo {
public
Data data;
// 信息
public
Integer error_code;
// 错误代码
public
String error_msg;
// 错误信息
public
static
class
Data {
public
String curVersion;
// 当前版本
public
String appURL;
// 下载地址
public
String description;
// 描述
public
String minVersion;
// 最低版本
public
String appName;
// 应用名称
}
@Override
public
String toString() {
return
"当前版本: "
+ data.curVersion +
", 下载地址: "
+ data.appURL +
", 描述信息: "
+ data.description
+
", 最低版本: "
+ data.minVersion +
", 应用代称: "
+ data.appName
+
", 错误代码: "
+ error_code +
", 错误信息: "
+ error_msg;
}
}
3. 请求和下载
更新库的主类, 包含
检查更新(checkUpdate)
和下载Apk(downloadApk)
两个重要方法.
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112/**
* 更新管理器
* <p>
* Created by wangchenlong on 16/1/6.
*/
@SuppressWarnings
(
"unused"
)
public
class
UpdateAppUtils {
@SuppressWarnings
(
"unused"
)
private
static
final
String TAG =
"DEBUG-WCL: "
+ UpdateAppUtils.
class
.getSimpleName();
/**
* 检查更新
*/
@SuppressWarnings
(
"unused"
)
public
static
void
checkUpdate(String appCode, String curVersion, UpdateCallback updateCallback) {
UpdateService updateService =
ServiceFactory.createServiceFrom(UpdateService.
class
, UpdateService.ENDPOINT);
updateService.getUpdateInfo(appCode, curVersion)
.subscribeOn(Schedulers.newThread())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(updateInfo -> onNext(updateInfo, updateCallback),
throwable -> onError(throwable, updateCallback));
}
// 显示信息
private
static
void
onNext(UpdateInfo updateInfo, UpdateCallback updateCallback) {
Log.e(TAG,
"返回数据: "
+ updateInfo.toString());
if
(updateInfo.error_code !=
0
|| updateInfo.data ==
null
||
updateInfo.data.appURL ==
null
) {
updateCallback.onError();
// 失败
}
else
{
updateCallback.onSuccess(updateInfo);
}
}
// 错误信息
private
static
void
onError(Throwable throwable, UpdateCallback updateCallback) {
updateCallback.onError();
}
/**
* 下载Apk, 并设置Apk地址,
* 默认位置: /storage/sdcard0/Download
*
* @param context 上下文
* @param updateInfo 更新信息
* @param infoName 通知名称
* @param storeApk 存储的Apk
*/
@SuppressWarnings
(
"unused"
)
public
static
void
downloadApk(
Context context, UpdateInfo updateInfo,
String infoName, String storeApk
) {
if
(!isDownloadManagerAvailable()) {
return
;
}
String description = updateInfo.data.description;
String appUrl = updateInfo.data.appURL;
if
(appUrl ==
null
|| appUrl.isEmpty()) {
Log.e(TAG,
"请填写\"App下载地址\""
);
return
;
}
appUrl = appUrl.trim();
// 去掉首尾空格
if
(!appUrl.startsWith(
"http"
)) {
}
Log.e(TAG,
"appUrl: "
+ appUrl);
DownloadManager.Request request;
try
{
request =
new
DownloadManager.Request(Uri.parse(appUrl));
}
catch
(Exception e) {
e.printStackTrace();
return
;
}
request.setTitle(infoName);
request.setDescription(description);
if
(Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
request.allowScanningByMediaScanner();
request.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED);
}
request.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, storeApk);
Context appContext = context.getApplicationContext();
DownloadManager manager = (DownloadManager)
appContext.getSystemService(Context.DOWNLOAD_SERVICE);
// 存储下载Key
SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(appContext);
sp.edit().putLong(PrefsConsts.DOWNLOAD_APK_ID_PREFS, manager.enqueue(request)).apply();
}
// 最小版本号大于9
private
static
boolean
isDownloadManagerAvailable() {
return
Build.VERSION.SDK_INT >= Build.VERSION_CODES.GINGERBREAD;
}
// 错误回调
public
interface
UpdateCallback {
void
onSuccess(UpdateInfo updateInfo);
void
onError();
}
}
检查更新: 创建服务, 在新线程中发送请求, 在主线程中接收数据, 判断成功和失败.
1234567891011121314/**
* 检查更新
*/
@SuppressWarnings
(
"unused"
)
public
static
void
checkUpdate(String appCode, String curVersion, UpdateCallback updateCallback) {
UpdateService updateService =
ServiceFactory.createServiceFrom(UpdateService.
class
, UpdateService.ENDPOINT);
updateService.getUpdateInfo(appCode, curVersion)
.subscribeOn(Schedulers.newThread())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(updateInfo -> onNext(updateInfo, updateCallback),
throwable -> onError(throwable, updateCallback));
}
下载Apk: 转换和解析Url, 设置通知信息和存储位置, 存储下载Id, 自动安装更新.
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657/**
* 下载Apk, 并设置Apk地址,
* 默认位置: /storage/sdcard0/Download
*
* @param context 上下文
* @param updateInfo 更新信息
* @param infoName 通知名称
* @param storeApk 存储的Apk
*/
@SuppressWarnings
(
"unused"
)
public
static
void
downloadApk(
Context context, UpdateInfo updateInfo,
String infoName, String storeApk
) {
if
(!isDownloadManagerAvailable()) {
return
;
}
String description = updateInfo.data.description;
String appUrl = updateInfo.data.appURL;
if
(appUrl ==
null
|| appUrl.isEmpty()) {
Log.e(TAG,
"请填写\"App下载地址\""
);
return
;
}
appUrl = appUrl.trim();
// 去掉首尾空格
if
(!appUrl.startsWith(
"http"
)) {
}
Log.e(TAG,
"appUrl: "
+ appUrl);
DownloadManager.Request request;
try
{
request =
new
DownloadManager.Request(Uri.parse(appUrl));
}
catch
(Exception e) {
e.printStackTrace();
return
;
}
request.setTitle(infoName);
request.setDescription(description);
if
(Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
request.allowScanningByMediaScanner();
request.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED);
}
request.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, storeApk);
Context appContext = context.getApplicationContext();
DownloadManager manager = (DownloadManager)
appContext.getSystemService(Context.DOWNLOAD_SERVICE);
// 存储下载Key
SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(appContext);
sp.edit().putLong(PrefsConsts.DOWNLOAD_APK_ID_PREFS, manager.enqueue(request)).apply();
}
使用
DownloadManager
下载文件是Android的推荐方式.
存储下载Id(manager.enqueue(request))
是为了在安装应用时, 找到Apk.
默认存储地址/storage/sdcard0/Download
.4.自动安装
注册广播接收器, 接收消息
ACTION_DOWNLOAD_COMPLETE
, 下载完成会发送广播. 获取下载文件的Uri, 进行匹配, 发送安装消息, 自动安装.
1234567891011121314151617181920212223242526272829303132333435363738/**
* 安装下载接收器
* <p>
* Created by wangchenlong on 16/1/5.
*/
public
class
InstallReceiver
extends
BroadcastReceiver {
private
static
final
String TAG =
"DEBUG-WCL: "
+ InstallReceiver.
class
.getSimpleName();
// 安装下载接收器
@Override
public
void
onReceive(Context context, Intent intent) {
if
(intent.getAction().equals(DownloadManager.ACTION_DOWNLOAD_COMPLETE)) {
long
downloadApkId = intent.getLongExtra(DownloadManager.EXTRA_DOWNLOAD_ID, -
1
);
installApk(context, downloadApkId);
}
}
// 安装Apk
private
void
installApk(Context context,
long
downloadApkId) {
// 获取存储ID
SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(context);
long
id = sp.getLong(PrefsConsts.DOWNLOAD_APK_ID_PREFS, -1L);
if
(downloadApkId == id) {
DownloadManager dManager = (DownloadManager) context.getSystemService(Context.DOWNLOAD_SERVICE);
Intent install =
new
Intent(Intent.ACTION_VIEW);
Uri downloadFileUri = dManager.getUriForDownloadedFile(downloadApkId);
if
(downloadFileUri !=
null
) {
install.setDataAndType(downloadFileUri,
"application/vnd.android.package-archive"
);
install.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(install);
}
else
{
Log.e(TAG,
"下载失败"
);
}
}
}
}
安装本应用下载的Apk, 不安装其他Apk, 存储下载Id, 与广播Id进行匹配.
下载失败, 也会发送下载完成(ACTION_DOWNLOAD_COMPLETE)
广播, Uri可能为空, 需要判断, 否则发生崩溃.
在应用中更新App版本的更多相关文章
-
iOS App Store上架新APP与更新APP版本
iOS App Store上架新APP与更新APP版本 http://www.jianshu.com/p/9e8d1edca148
-
更新App版本的流程
上班一年了还没有自己打包上传过APP,周五下班时项目经理手把手教了我一遍,我大致把流程在这里回顾一下: 1.首先要将svn上的代码拷贝一份到分支上,用终端操作:svn cp https://192.1 ...
-
在应用中更新App版本号
在应用中, 为了提高用户体验, 会提供更新版本号的功能. 那么怎样实现呢? 我写了一个简单的Demo, 说明一下, 须要注意几个细节. 使用了Retrofit和Rx处理网络请求. Github下载地址 ...
-
【ionic App问题总结系列】ionic 如何更新app版本
ionic 如何进行自动更新 ionic App更新有两种方式:第一种是普通的从远程下载apk,安装并覆盖旧版本.另外一种就是采用替换www文件夹的内容,实现应用内更新,而无需下载安装apk. 这篇文 ...
-
apicloud 上传/更新App版本到 ios store 流程步骤
app更新 上传APP的地址: https://itunesconnect.apple.com/login 苹果开发者中心: https://developer.apple.com/ app正式包更新 ...
-
iOS开发 判断当前APP版本和升级
从iOS8系统开始,用户可以在设置里面设置在WiFi环境下,自动更新安装的App.此功能大大方便了用户,但是一些用户没有开启此项功能,因此还是需要在程序里面提示用户的 方法一:在服务器接口约定对应的数 ...
-
iOS中如何知道app版本已更新
主要用于程序升级,开启程序后是否显示新特性两个方面. 1.苹果app版本 苹果规定,程序的版本只能升不能降.例如1.0->1.1可以,1.1->1.0就不可以,不允许上架. 2.app版本 ...
-
iOS开发之一句代码检测APP版本的更新
提示更新效果图如下,当然也是可以自定义类似与AlertView相似的自定义view,如京东.网易云音乐都是自定义了这种提示框的view.以下只展示,从App Store获取到app信息.并解析app信 ...
-
Android实现App版本自动更新
现在很多的App中都会有一个检查版本的功能.例如斗鱼TV App的设置界面下: 当我们点击检查更新的时候,就会向服务器发起版本检测的请求.一般的处理方式是:服务器返回的App版本与当前手机安装的版本号 ...
随机推荐
-
mysql查询中通配符的使用
mysql查询中通配符的使用 在mysql查询中经常会使用通配符,并且mysql的通配符和pgsql的存在区别(稍候再讨论),而且mysql中还可以使用正则表达式. SQL模式匹配: “_” ...
-
Windows8 10设置程序为 系统默认浏览器
从win8 开始,MS修改了文件和协议的关联方式,普通的注册表修改是无效的. 必须使用组策略(group policy )对象GP才行. http://blogs.technet.com/b/mrml ...
-
1.解剖Linq to object
LINQ想必大家都不陌生了,它的出现使得我们的代码变得更短.更优雅了.至于LINQ是什么,Linq to object这类的扩展方法到底做了些什么.我们使用的EF是如何实现的(如何解析Expressi ...
-
Oracle 11g+oracle客户端(32位)+PL/SQL develepment的安装配置
之前一直想学Oracle,可是就是安装配置Oracle一直未成功,让人很苦恼,特别是什么监听器什么的,一直没搞明白,弄了整整一天都没弄出来,上网查资料后发现资料上大多数都是参差不齐,不太详细明了,尝试 ...
-
全球排名第一的免费开源ERP Odoo 12产品发布会北京站开始报名
Odoo V12 产品(北京)发布会 暨企业数字化转型论坛 快速报名通道:http://odoochina.mikecrm.com/uG8nNu4 随着新版本Odoo 12的发布,开源智造(OSCG. ...
-
Postman应用笔记
Postman应用: 项目组织格式 Collections 集合--项目--根路径文件夹 文件夹 集合下只支持1级文件夹 文件夹 Request --请求 url 认证参数,头信息,体信息(Autho ...
-
java快速排序引起的*Error异常
写在前面:这篇随笔主要记录一下递归调用引起的虚拟机栈溢出的情况以及通过参数配置了虚拟机栈大小来使递归调用可以顺利执行.并没有对涉及到的一些概念进行详细的解释(因为我自己目前对这些概念并不是特别清楚), ...
-
版本控制 version control和团队协作
这些技术你可能暂时不会用到,但是一旦软件体量变大,开发人数增加,这就带来质变,需要借助一些工具或者技术才能完成这些复杂的工程. 你可以从最简单的情况思考,你可以对任何类型的文件进行版本控制,比如一个p ...
-
mysql (主从复制)(proxy , Amoeba)
原址如下: http://heylinux.com/archives/1004.html Mysql作为目前世界上使用最广泛的免费数据库,相信所有从事系统运维的工程师都一定接触过.但在实际的生产环境中 ...
-
20155323 2016-2017-2 《Java程序设计》第10周学习总结
20155323 2016-2017-2 <Java程序设计>第10周学习总结 教材学习内容总结 网络编程就是在两个或两个以上的设备(例如计算机)之间传输数据. 狭义的网络编程范畴就是程序 ...