實(shí)現(xiàn)應(yīng)用內(nèi)的更新版本功能

歡迎Follow我的GitHub, 關(guān)注我的簡(jiǎn)書. 其余參考Android目錄.

更新

本文的合集已經(jīng)編著成書蚂子,高級(jí)Android開發(fā)強(qiáng)化實(shí)戰(zhàn)叉趣,歡迎各位讀友的建議和指導(dǎo)艳丛。在京東即可購(gòu)買:https://item.jd.com/12385680.html

Android

在應(yīng)用中, 為了提高用戶體驗(yàn), 會(huì)提供更新版本的功能. 那么如何實(shí)現(xiàn)呢? 我寫了一個(gè)簡(jiǎn)單的Demo, 說明一下, 需要注意幾個(gè)細(xì)節(jié). 使用了Retrofit和Rx處理網(wǎng)絡(luò)請(qǐng)求.

本文源碼的GitHub下載地址

1. 邏輯

訪問服務(wù)器, 根據(jù)是否包含新版本, 判斷是否需要更新.
下載Apk, 下載完成后, 自動(dòng)安裝, 高版本會(huì)覆蓋低版本.

邏輯:

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 = "計(jì)步器";
    private static final String STORE_APK = "chunyu_apk";

    @Bind(R.id.main_b_install_apk) Button mBInstallApk;

    private UpdateAppUtils.UpdateCallback mUpdateCallback; // 更新回調(diào)

    @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的代號(hào), 版本號(hào), 異步回調(diào), 發(fā)送到服務(wù)器, 判斷是否需要更新. 如果存在新版本, 則下載Apk, 并自動(dòng)安裝更新.

2. 網(wǎng)絡(luò)請(qǐng)求

更新請(qǐng)求, 參數(shù)是App代號(hào)和當(dāng)前版本號(hào).

/**
 * 更新服務(wù)
 * <p>
 * Created by wangchenlong on 16/1/4.
 */
public interface UpdateService {
    String ENDPOINT = "http://www.chunyuyisheng.com";

    // 獲取個(gè)人信息
    @GET("/cmsapi/app/update")
    Observable<UpdateInfo> getUpdateInfo(
            @Query("appName") String appName,
            @Query("version") String version);
}

創(chuàng)建服務(wù)的工廠類.

/**
 * 創(chuàng)建Retrofit服務(wù)
 * <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轉(zhuǎn)換器
                .build();
        return adapter.create(serviceClass);
    }
}

更新信息的Json類.

/**
 * 更新信息(JSON)
 * <p>
 * Created by wangchenlong on 16/1/4.
 */
public class UpdateInfo {
    public Data data; // 信息
    public Integer error_code; // 錯(cuò)誤代碼
    public String error_msg; // 錯(cuò)誤信息

    public static class Data {
        public String curVersion; // 當(dāng)前版本
        public String appURL; // 下載地址
        public String description; // 描述
        public String minVersion; // 最低版本
        public String appName; // 應(yīng)用名稱
    }

    @Override public String toString() {
        return "當(dāng)前版本: " + data.curVersion + ", 下載地址: " + data.appURL + ", 描述信息: " + data.description
                + ", 最低版本: " + data.minVersion + ", 應(yīng)用代稱: " + data.appName
                + ", 錯(cuò)誤代碼: " + error_code + ", 錯(cuò)誤信息: " + error_msg;
    }
}

3. 請(qǐng)求和下載

更新庫的主類, 包含檢查更新(checkUpdate)下載Apk(downloadApk)兩個(gè)重要方法.

/**
 * 更新管理器
 * <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, "返回?cái)?shù)據(jù): " + updateInfo.toString());
        if (updateInfo.error_code != 0 || updateInfo.data == null ||
                updateInfo.data.appURL == null) {
            updateCallback.onError(); // 失敗
        } else {
            updateCallback.onSuccess(updateInfo);
        }
    }

    // 錯(cuò)誤信息
    private static void onError(Throwable throwable, UpdateCallback updateCallback) {
        updateCallback.onError();
    }

    /**
     * 下載Apk, 并設(shè)置Apk地址,
     * 默認(rèn)位置: /storage/sdcard0/Download
     *
     * @param context    上下文
     * @param updateInfo 更新信息
     * @param infoName   通知名稱
     * @param storeApk   存儲(chǔ)的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, "請(qǐng)?zhí)顚慭"App下載地址\"");
            return;
        }

        appUrl = appUrl.trim(); // 去掉首尾空格

        if (!appUrl.startsWith("http")) {
            appUrl = "http://" + appUrl; // 添加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);

        // 存儲(chǔ)下載Key
        SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(appContext);
        sp.edit().putLong(PrefsConsts.DOWNLOAD_APK_ID_PREFS, manager.enqueue(request)).apply();
    }

    // 最小版本號(hào)大于9
    private static boolean isDownloadManagerAvailable() {
        return Build.VERSION.SDK_INT >= Build.VERSION_CODES.GINGERBREAD;
    }

    // 錯(cuò)誤回調(diào)
    public interface UpdateCallback {
        void onSuccess(UpdateInfo updateInfo);

        void onError();
    }
}

檢查更新: 創(chuàng)建服務(wù), 在新線程中發(fā)送請(qǐng)求, 在主線程中接收數(shù)據(jù), 判斷成功和失敗.

    /**
     * 檢查更新
     */
    @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: 轉(zhuǎn)換和解析Url, 設(shè)置通知信息和存儲(chǔ)位置, 存儲(chǔ)下載Id, 自動(dòng)安裝更新.

    /**
     * 下載Apk, 并設(shè)置Apk地址,
     * 默認(rèn)位置: /storage/sdcard0/Download
     *
     * @param context    上下文
     * @param updateInfo 更新信息
     * @param infoName   通知名稱
     * @param storeApk   存儲(chǔ)的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, "請(qǐng)?zhí)顚慭"App下載地址\"");
            return;
        }

        appUrl = appUrl.trim(); // 去掉首尾空格

        if (!appUrl.startsWith("http")) {
            appUrl = "http://" + appUrl; // 添加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);

        // 存儲(chǔ)下載Key
        SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(appContext);
        sp.edit().putLong(PrefsConsts.DOWNLOAD_APK_ID_PREFS, manager.enqueue(request)).apply();
    }

使用DownloadManager下載文件是Android的推薦方式.
存儲(chǔ)下載Id(manager.enqueue(request))是為了在安裝應(yīng)用時(shí), 找到Apk.
默認(rèn)存儲(chǔ)地址/storage/sdcard0/Download.

4.自動(dòng)安裝

注冊(cè)廣播接收器, 接收消息ACTION_DOWNLOAD_COMPLETE, 下載完成會(huì)發(fā)送廣播. 獲取下載文件的Uri, 進(jìn)行匹配, 發(fā)送安裝消息, 自動(dòng)安裝.

/**
 * 安裝下載接收器
 * <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) {
        // 獲取存儲(chǔ)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, "下載失敗");
            }
        }
    }
}

安裝本應(yīng)用下載的Apk, 不安裝其他Apk, 存儲(chǔ)下載Id, 與廣播Id進(jìn)行匹配.
下載失敗, 也會(huì)發(fā)送下載完成(ACTION_DOWNLOAD_COMPLETE)廣播, Uri可能為空, 需要判斷, 否則發(fā)生崩潰.

OK, that's all! Enjoy It!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市有额,隨后出現(xiàn)的幾起案子竹海,更是在濱河造成了極大的恐慌慕蔚,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,858評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件斋配,死亡現(xiàn)場(chǎng)離奇詭異孔飒,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)艰争,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,372評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門坏瞄,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人甩卓,你說我怎么就攤上這事鸠匀。” “怎么了逾柿?”我有些...
    開封第一講書人閱讀 165,282評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵缀棍,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我鹿寻,道長(zhǎng)睦柴,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,842評(píng)論 1 295
  • 正文 為了忘掉前任毡熏,我火速辦了婚禮坦敌,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己狱窘,他們只是感情好杜顺,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,857評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著蘸炸,像睡著了一般躬络。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上搭儒,一...
    開封第一講書人閱讀 51,679評(píng)論 1 305
  • 那天穷当,我揣著相機(jī)與錄音,去河邊找鬼淹禾。 笑死馁菜,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的铃岔。 我是一名探鬼主播汪疮,決...
    沈念sama閱讀 40,406評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼毁习!你這毒婦竟也來了智嚷?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,311評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤纺且,失蹤者是張志新(化名)和其女友劉穎盏道,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體载碌,經(jīng)...
    沈念sama閱讀 45,767評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡摇天,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,945評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了恐仑。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,090評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡为鳄,死狀恐怖裳仆,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情孤钦,我是刑警寧澤歧斟,帶...
    沈念sama閱讀 35,785評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站偏形,受9級(jí)特大地震影響静袖,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜俊扭,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,420評(píng)論 3 331
  • 文/蒙蒙 一队橙、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦捐康、人聲如沸仇矾。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,988評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽贮匕。三九已至,卻和暖如春花枫,著一層夾襖步出監(jiān)牢的瞬間刻盐,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,101評(píng)論 1 271
  • 我被黑心中介騙來泰國(guó)打工劳翰, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留敦锌,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,298評(píng)論 3 372
  • 正文 我出身青樓磕道,卻偏偏與公主長(zhǎng)得像供屉,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子溺蕉,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,033評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理伶丐,服務(wù)發(fā)現(xiàn),斷路器疯特,智...
    卡卡羅2017閱讀 134,659評(píng)論 18 139
  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,166評(píng)論 25 707
  • ¥開啟¥ 【iAPP實(shí)現(xiàn)進(jìn)入界面執(zhí)行逐一顯】 〖2017-08-25 15:22:14〗 《//首先開一個(gè)線程哗魂,因...
    小菜c閱讀 6,424評(píng)論 0 17
  • Read through J. k Rowling's speech at Harvard on 2008, it...
    Lifefullofjoy閱讀 165評(píng)論 2 1
  • 不可努力原理:今天我得幫那些不主動(dòng)努力的人們開脫一把[調(diào)皮](此處應(yīng)有掌聲)。之前提到個(gè)生態(tài)的穩(wěn)定性漓雅,這里的定義準(zhǔn)...
    問基閱讀 577評(píng)論 0 0