在应用中更新App版本
在应用中, 为了提高用户体验, 会提供更新版本的功能. 那么如何实现呢? 我写了一个简单的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, 下载完成后, 自动安装, 高版本会覆盖低版本.
逻辑:
1234567891011121314151617181920212223242526272829303132333435publicclassMainActivityextendsAppCompatActivity {privatestaticfinalString APP_NAME ="Ped_android";privatestaticfinalString VERSION ="1.0.0";privatestaticfinalString INFO_NAME ="计步器";privatestaticfinalString STORE_APK ="chunyu_apk";@Bind(R.id.main_b_install_apk) Button mBInstallApk;privateUpdateAppUtils.UpdateCallback mUpdateCallback;// 更新回调@OverrideprotectedvoidonCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);ButterKnife.bind(this);mUpdateCallback =newUpdateAppUtils.UpdateCallback() {@OverridepublicvoidonSuccess(UpdateInfo updateInfo) {Toast.makeText(MainActivity.this,"有更新", Toast.LENGTH_SHORT).show();UpdateAppUtils.downloadApk(MainActivity.this, updateInfo, INFO_NAME, STORE_APK);}@OverridepublicvoidonError() {Toast.makeText(MainActivity.this,"无更新", Toast.LENGTH_SHORT).show();}};mBInstallApk.setOnClickListener(newView.OnClickListener() {@OverridepublicvoidonClick(View v) {UpdateAppUtils.checkUpdate(APP_NAME, VERSION, mUpdateCallback);}});}}
UpdateAppUtils是核心下载类. 输入App的代号, 版本号, 异步回调, 发送到服务器, 判断是否需要更新. 如果存在新版本, 则下载Apk, 并自动安装更新.2. 网络请求
更新请求, 参数是App代号和当前版本号.
1234567891011121314/*** 更新服务* <p>* Created by wangchenlong on 16/1/4.*/publicinterfaceUpdateService {// 获取个人信息@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.*/publicclassServiceFactory {publicstatic<T> T createServiceFrom(finalClass<T> serviceClass, String endpoint) {Retrofit adapter =newRetrofit.Builder().baseUrl(endpoint).addCallAdapterFactory(RxJavaCallAdapterFactory.create())// 添加Rx适配器.addConverterFactory(GsonConverterFactory.create())// 添加Gson转换器.build();returnadapter.create(serviceClass);}}更新信息的Json类.
123456789101112131415161718192021222324/*** 更新信息(JSON)* <p>* Created by wangchenlong on 16/1/4.*/publicclassUpdateInfo {publicData data;// 信息publicInteger error_code;// 错误代码publicString error_msg;// 错误信息publicstaticclassData {publicString curVersion;// 当前版本publicString appURL;// 下载地址publicString description;// 描述publicString minVersion;// 最低版本publicString appName;// 应用名称}@OverridepublicString 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")publicclassUpdateAppUtils {@SuppressWarnings("unused")privatestaticfinalString TAG ="DEBUG-WCL: "+ UpdateAppUtils.class.getSimpleName();/*** 检查更新*/@SuppressWarnings("unused")publicstaticvoidcheckUpdate(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));}// 显示信息privatestaticvoidonNext(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);}}// 错误信息privatestaticvoidonError(Throwable throwable, UpdateCallback updateCallback) {updateCallback.onError();}/*** 下载Apk, 并设置Apk地址,* 默认位置: /storage/sdcard0/Download** @param context 上下文* @param updateInfo 更新信息* @param infoName 通知名称* @param storeApk 存储的Apk*/@SuppressWarnings("unused")publicstaticvoiddownloadApk(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 =newDownloadManager.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);// 存储下载KeySharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(appContext);sp.edit().putLong(PrefsConsts.DOWNLOAD_APK_ID_PREFS, manager.enqueue(request)).apply();}// 最小版本号大于9privatestaticbooleanisDownloadManagerAvailable() {returnBuild.VERSION.SDK_INT >= Build.VERSION_CODES.GINGERBREAD;}// 错误回调publicinterfaceUpdateCallback {voidonSuccess(UpdateInfo updateInfo);voidonError();}}检查更新: 创建服务, 在新线程中发送请求, 在主线程中接收数据, 判断成功和失败.
1234567891011121314/*** 检查更新*/@SuppressWarnings("unused")publicstaticvoidcheckUpdate(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")publicstaticvoiddownloadApk(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 =newDownloadManager.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);// 存储下载KeySharedPreferences 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.*/publicclassInstallReceiverextendsBroadcastReceiver {privatestaticfinalString TAG ="DEBUG-WCL: "+ InstallReceiver.class.getSimpleName();// 安装下载接收器@OverridepublicvoidonReceive(Context context, Intent intent) {if(intent.getAction().equals(DownloadManager.ACTION_DOWNLOAD_COMPLETE)) {longdownloadApkId = intent.getLongExtra(DownloadManager.EXTRA_DOWNLOAD_ID, -1);installApk(context, downloadApkId);}}// 安装ApkprivatevoidinstallApk(Context context,longdownloadApkId) {// 获取存储IDSharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(context);longid = sp.getLong(PrefsConsts.DOWNLOAD_APK_ID_PREFS, -1L);if(downloadApkId == id) {DownloadManager dManager = (DownloadManager) context.getSystemService(Context.DOWNLOAD_SERVICE);Intent install =newIntent(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版本与当前手机安装的版本号 ...
随机推荐
- hibernate有关联关系删除子表时可能会报错,可以用个clear避免错误
//清除子表数据 public SalesSet removeSalesSetDistributor(SalesSet salesSet ){ List<SalesSetDistributor& ...
- C++导出文件后缀dll,lib,exp,def
简要的介绍一下在微软开发工具中(VC)静态链接库和动态链接库生成过程中出现的.dll .lib .def 和 .exp文件类型.windows平台上可执行文件可能是一个.exe文件也可能四个.dl ...
- 2016年12月16日 星期五 --出埃及记 Exodus 21:11
2016年12月16日 星期五 --出埃及记 Exodus 21:11 If he does not provide her with these three things, she is to go ...
- 测可用!ecshop立即购买和加入购物车按钮共存的方法
网上方法很多,但都不能用的,有的是老版本的,有的方法本身就不完整. 应大多数客户要求,我们重新整理下教程,希望对大家有用. 亲测可用!ecshop立即购买和加入购物车按钮共存的方法 第一步:修改ecs ...
- C语言培训第一天
下面是一些命令,先来谈谈今天的若干收获吧! 计算机中的一切文件都是以二进制补码的形式存在,问题也就来了. 第一个问题 如果我们给一个无符号的数赋值一个负数,他会读取到什么,又会输出什么?(似乎问题和上 ...
- UE4编程之C++创建一个FPS工程(一)创建模式&角色&处理输入
转自:http://blog.csdn.net/u011707076/article/details/44180951 从今天开始,我们一起来学习一下,如何使用C++将一个不带有任何初学者内容的空模板 ...
- 搭建本地的git仓库
折腾了快一天了,终于搭建成功了. 分享一下搭建的步骤: 一.GIT仓库的创建 1. adduser git 2. passwd git 此例设置git的密码为123456 3. cd /home/gi ...
- json和jsonp的传输方式
jsonp传输会解决跨域的问题 $.ajax({ async: false, /* url: "http://127.0.0.1:8080/2015020601/background/mea ...
- 5.5.3使用terminfo功能标志
当使用terminfo时,需要做的第一件事就是调用函数setupterm来设置终端类型.这将为当前的终端类型初始化一个TERMINAL结构.然后,你就可以查看当前终端的功能标志并使用他们的功能了. # ...
- 5.mybatis一对一表关联查询
方式一:嵌套结果:使用嵌套结果映射来处理重复的联合结果的子集,封装联表查询的数据(去除重复的数据) SELECT * FROM class c,teacher t WHERE c.tid = t.t ...