Android實(shí)現(xiàn)APP在線下載更新
前言
項(xiàng)目地址:UpdateAppDemo
現(xiàn)在的android應(yīng)用app會(huì)隔一段時(shí)間發(fā)布一個(gè)新的版本,當(dāng)你打開(kāi)某個(gè)app,如果有最新的版本,會(huì)提醒你是否下載更新。本文利用android自帶的下載管理器DownloadManager進(jìn)行下載最新版本的apk,下載完成后自動(dòng)跳轉(zhuǎn)安裝。效果如下:

第一步、檢查版本并判斷是否需要更新
通過(guò)獲取當(dāng)前app版本號(hào)與服務(wù)器上的版本號(hào)進(jìn)行對(duì)比,如果本地的版本號(hào)低于服務(wù)器版本號(hào),則彈出提示框:發(fā)現(xiàn)新版本,是否下載更新。
/**
* Created by Teprinciple on 2016/11/15.
*/
public class UpdateAppUtil {
/**
* 獲取當(dāng)前apk的版本號(hào) currentVersionCode
* @param ctx
* @return
*/
public static int getAPPLocalVersion(Context ctx) {
int currentVersionCode = 0;
PackageManager manager = ctx.getPackageManager();
try {
PackageInfo info = manager.getPackageInfo(ctx.getPackageName(), 0);
String appVersionName = info.versionName; // 版本名
currentVersionCode = info.versionCode; // 版本號(hào)
} catch (PackageManager.NameNotFoundException e) {
e.printStackTrace();
}
return currentVersionCode;
}
/**
* 獲取服務(wù)器上版本信息
* @param context
* @param callBack
*/
public static void getAPPServerVersion(Context context, final VersionCallBack callBack){
HttpUtil.getObject(Api.GETVERSION.mapClear().addBody(), VersionInfo.class, new HttpUtil.ObjectCallback() {
@Override
public void result(boolean b, @Nullable Object obj) {
if (b){
callBack.callBack((VersionInfo) obj);
}
}
});
}
/**
* 判斷版本號(hào),更新APP
* @param context
*/
public static void updateApp(final Context context){
getAPPServerVersion(context, new VersionCallBack() {
@Override
public void callBack(final VersionInfo info) {
if (info != null && info.getVersionCode()!=null){
Log.i("this","版本信息:當(dāng)前"+getAPPLocalVersion(context)+",服務(wù)器:"+Integer.valueOf(info.getVersionCode()));
if (Integer.valueOf(info.getVersionCode()) > getAPPLocalVersion(context)){
ConfirmDialog dialog = new ConfirmDialog(context, new lht.wangtong.gowin120.doctor.views.feature.Callback() {
@Override
public void callback() {
DownloadAppUtils.downloadForAutoInstall(context, Api.HOST_IMG+info.getLoadPath(), "demo.apk", "更新demo");
}
});
dialog .setContent("發(fā)現(xiàn)新版本:"+info.getVersionNumber()+"\n是否下載更新?");
dialog.setCancelable(false);
dialog .show();
}
}
}
});
}
public interface VersionCallBack{
void callBack(VersionInfo info);
}
}
第二步、下載最新版apk
通過(guò)Android自帶的DownloadManager下載管理器,下載服務(wù)器上最新版的apk。下載完成后會(huì)發(fā)送下載完成的廣播。
/**
* Created by Teprinciple on 2016/11/15.
*/
public class DownloadAppUtils {
private static final String TAG = DownloadAppUtils.class.getSimpleName();
public static long downloadUpdateApkId = -1;//下載更新Apk 下載任務(wù)對(duì)應(yīng)的Id
public static String downloadUpdateApkFilePath;//下載更新Apk 文件路徑
/**
* 通過(guò)瀏覽器下載APK包
* @param context
* @param url
*/
public static void downloadForWebView(Context context, String url) {
Uri uri = Uri.parse(url);
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setDataAndType(Uri.fromFile(new File(Environment
.getExternalStorageDirectory(), "tmp.apk")),
"application/vnd.android.package-archive");
context.startActivity(intent);
}
/**
* 下載更新apk包
* 權(quán)限:1,<uses-permission android:name="android.permission.DOWNLOAD_WITHOUT_NOTIFICATION" />
* @param context
* @param url
*/
public static void downloadForAutoInstall(Context context, String url, String fileName, String title) {
//LogUtil.e("App 下載 url="+url+",fileName="+fileName+",title="+title);
if (TextUtils.isEmpty(url)) {
return;
}
try {
Uri uri = Uri.parse(url);
DownloadManager downloadManager = (DownloadManager) context
.getSystemService(Context.DOWNLOAD_SERVICE);
DownloadManager.Request request = new DownloadManager.Request(uri);
//在通知欄中顯示
request.setVisibleInDownloadsUi(true);
request.setTitle(title);
String filePath = null;
if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {//外部存儲(chǔ)卡
filePath = Environment.getExternalStorageDirectory().getAbsolutePath();
} else {
T.showShort(context, R.string.download_sdcard_error);
return;
}
downloadUpdateApkFilePath = filePath + File.separator + fileName;
// 若存在,則刪除
deleteFile(downloadUpdateApkFilePath);
Uri fileUri = Uri.parse("file://" + downloadUpdateApkFilePath);
request.setDestinationUri(fileUri);
downloadUpdateApkId = downloadManager.enqueue(request);
} catch (Exception e) {
e.printStackTrace();
downloadForWebView(context, url);
}
}
private static boolean deleteFile(String fileStr) {
File file = new File(fileStr);
return file.delete();
}
}
注意添加權(quán)限:
<uses-permission android:name="android.permission.DOWNLOAD_WITHOUT_NOTIFICATION" />
第三步、下載完成后跳轉(zhuǎn)安裝
通過(guò)廣播接收者,接收到下載完成后發(fā)出的廣播,跳轉(zhuǎn)到系統(tǒng)的安裝界面,進(jìn)行安裝。
/**
* Created by Teprinciple on 2016/11/15.
*/
public class UpdateAppReceiver extends BroadcastReceiver {
public UpdateAppReceiver() {
}
@Override
public void onReceive(Context context, Intent intent) {
// 處理下載完成
Cursor c=null;
try {
if (DownloadManager.ACTION_DOWNLOAD_COMPLETE.equals(intent.getAction())) {
if (DownloadAppUtils.downloadUpdateApkId >= 0) {
long downloadId = DownloadAppUtils.downloadUpdateApkId;
DownloadManager.Query query = new DownloadManager.Query();
query.setFilterById(downloadId);
DownloadManager downloadManager = (DownloadManager) context
.getSystemService(Context.DOWNLOAD_SERVICE);
c = downloadManager.query(query);
if (c.moveToFirst()) {
int status = c.getInt(c
.getColumnIndex(DownloadManager.COLUMN_STATUS));
if (status == DownloadManager.STATUS_FAILED) {
downloadManager.remove(downloadId);
} else if (status == DownloadManager.STATUS_SUCCESSFUL) {
if (DownloadAppUtils.downloadUpdateApkFilePath != null) {
Intent i = new Intent(Intent.ACTION_VIEW);
i.setDataAndType(
Uri.parse("file://"
+ DownloadAppUtils.downloadUpdateApkFilePath),
"application/vnd.android.package-archive");
i.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(i);
}
}
}
}
}/* else if (DownloadManager.ACTION_NOTIFICATION_CLICKED.equals(intent.getAction())) {//點(diǎn)擊通知取消下載
DownloadManager downloadManager = (DownloadManager) context
.getSystemService(Context.DOWNLOAD_SERVICE);
long[] ids = intent.getLongArrayExtra(DownloadManager.EXTRA_NOTIFICATION_CLICK_DOWNLOAD_IDS);
//點(diǎn)擊通知欄取消下載
downloadManager.remove(ids);
}*/
} catch (Exception e) {
e.printStackTrace();
}finally {
if (c != null) {
c.close();
}
}
}
}
注意需要在AndroidMainfest.xml中注冊(cè)receiver:
<receiver android:name=".updateapp.UpdateAppReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.DOWNLOAD_COMPLETE" />
<action android:name="android.intent.action.DOWNLOAD_NOTIFICATION_CLICKED"/>
</intent-filter>
</receiver>
通過(guò)上面三步就可以快速實(shí)現(xiàn)APP的在線更新 。
項(xiàng)目地址:UpdateAppDemo
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- 基于Retrofit2+RxJava2實(shí)現(xiàn)Android App自動(dòng)更新
- android中強(qiáng)制更新app實(shí)例代碼
- Android實(shí)現(xiàn)簡(jiǎn)潔的APP更新dialog數(shù)字進(jìn)度條
- Android應(yīng)用App更新實(shí)例詳解
- Android應(yīng)用APP自動(dòng)更新功能的代碼實(shí)現(xiàn)
- Android App增量更新詳解及實(shí)例代碼
- Android如何實(shí)現(xiàn)APP自動(dòng)更新
- Android App實(shí)現(xiàn)應(yīng)用內(nèi)部自動(dòng)更新的最基本方法示例
- android實(shí)現(xiàn)通知欄下載更新app示例
- Android實(shí)現(xiàn)APP自動(dòng)更新功能
相關(guān)文章
另外兩種Android沉浸式狀態(tài)欄實(shí)現(xiàn)思路
這篇文章主要為大家介紹了另外兩種Android沉浸式狀態(tài)欄實(shí)現(xiàn)思路,android5.0及以后版本都支持給狀態(tài)欄著色,而目前android主流版本還是4.4,想要深入了解的朋友可以參考一下2016-01-01
Android利用Fragment實(shí)現(xiàn)Tab選項(xiàng)卡效果
這篇文章主要為大家詳細(xì)介紹了Android利用Fragment實(shí)現(xiàn)Tab選項(xiàng)卡效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-08-08
Android中SurfaceView和普通view的區(qū)別及使用
SurfaceView第一印象它是一個(gè)view,因?yàn)樗^承了View,本文主要介紹了SurfaceView和普通view的區(qū)別及使用,感興趣的小伙伴們可以參考一下2021-06-06
Android中基于XMPP協(xié)議實(shí)現(xiàn)IM聊天程序與多人聊天室
這篇文章主要介紹了Android中基于XMPP協(xié)議實(shí)現(xiàn)IM聊天程序與多人聊天室的方法,XMPP基于XML數(shù)據(jù)格式傳輸,一般用于即時(shí)消息(IM)以及在線現(xiàn)場(chǎng)探測(cè),需要的朋友可以參考下2016-02-02
Android簡(jiǎn)單實(shí)現(xiàn)app每月簽到功能
這篇文章主要為大家詳細(xì)介紹了Android簡(jiǎn)單實(shí)現(xiàn)app每月簽到功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-11-11
Android實(shí)現(xiàn)獲取短信驗(yàn)證碼并自動(dòng)填寫(xiě)功能
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)獲取短信驗(yàn)證碼并自動(dòng)填寫(xiě)功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-07-07
Android實(shí)現(xiàn)類(lèi)似微信的文本輸入框 效果
本文給大家介紹一下微信的文本輸入框是如何實(shí)現(xiàn)的,其實(shí)那只是個(gè)普通的文本框設(shè)了一個(gè)特殊的背景而已。具體微信怎么實(shí)現(xiàn)的,大家可以反編譯下,這里介紹下如何實(shí)現(xiàn)這個(gè)背景2017-05-05

