android 在线升级借助开源中国App源码
android 在线升级借助开源中国App源码
http://www.cnblogs.com/luomingui/p/3949429.html
android 在线升级借助开源中国App源码分析如下:
1: checkAppUpdate 检查是或需要升级
// 网络连接判断 if (appContext.isNetworkConnected()) { // 检查新版本 if (appContext.isCheckUp()) { UpdateManager.getUpdateManager().checkAppUpdate(this, false); } }
2: UpdateManager类
package lcl.android.core;
import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.net.HttpURLConnection; import java.net.MalformedURLException; import java.net.URL; import java.text.DecimalFormat;
import lcl.android.R; import lcl.android.api.ApiClient; import android.app.AlertDialog; import android.app.AlertDialog.Builder; import android.app.Dialog; import android.app.ProgressDialog; import android.content.Context; import android.content.DialogInterface; import android.content.DialogInterface.OnCancelListener; import android.content.DialogInterface.OnClickListener; import android.content.Intent; import android.content.pm.PackageInfo; import android.content.pm.PackageManager.NameNotFoundException; import android.net.Uri; import android.os.Environment; import android.os.Handler; import android.os.Message; import android.view.LayoutInflater; import android.view.View; import android.widget.ProgressBar; import android.widget.TextView; import android.widget.Toast;
/** * 应用程序更新工具包 * */ public class UpdateManager { private static final int DOWN_NOSDCARD = 0; private static final int DOWN_UPDATE = 1; private static final int DOWN_OVER = 2; private static final int DIALOG_TYPE_LATEST = 0; private static final int DIALOG_TYPE_FAIL = 1;
private static UpdateManager updateManager;
private Context mContext; // 通知对话框 private Dialog noticeDialog; // 下载对话框 private Dialog downloadDialog; // '已经是最新' 或者 '无法获取最新版本' 的对话框 private Dialog latestOrFailDialog; // 进度条 private ProgressBar mProgress; // 显示下载数值 private TextView mProgressText; // 查询动画 private ProgressDialog mProDialog; // 进度值 private int progress; // 下载线程 private Thread downLoadThread; // 终止标记 private boolean interceptFlag; // 提示语 private String updateMsg = ""; // 返回的安装包url private String apkUrl = ""; // 下载包保存路径 private String savePath = ""; // apk保存完整路径 private String apkFilePath = ""; // 临时下载文件路径 private String tmpFilePath = ""; // 下载文件大小 private String apkFileSize; // 已下载文件大小 private String tmpFileSize;
private String curVersionName = ""; private int curVersionCode; private Update mUpdate;
//单利模式
public static UpdateManager getUpdateManager() { if (updateManager == null) { updateManager = new UpdateManager(); } updateManager.interceptFlag = false; return updateManager; }
private Handler mHandler = new Handler() { public void handleMessage(Message msg) { switch (msg.what) { case DOWN_UPDATE: mProgress.setProgress(progress); mProgressText.setText(tmpFileSize + "/" + apkFileSize); break; case DOWN_OVER: downloadDialog.dismiss(); installApk(); break; case DOWN_NOSDCARD: downloadDialog.dismiss(); Toast.makeText(mContext, "无法下载安装文件,请检查SD卡是否挂载", 3000).show(); break; } }; };
/** * 获取当前客户端版本信息 */ private void getCurrentVersion() { try { PackageInfo info = mContext.getPackageManager().getPackageInfo( mContext.getPackageName(), 0); curVersionName = info.versionName; curVersionCode = info.versionCode; } catch (NameNotFoundException e) { e.printStackTrace(System.err); } }
/** * 检查App更新 * * @param context * @param isShowMsg * 是否显示提示消息 */ public void checkAppUpdate(Context context, final boolean isShowMsg) { this.mContext = context; getCurrentVersion(); if (isShowMsg) { if (mProDialog == null) mProDialog = ProgressDialog.show(mContext, null, "正在检测,请稍后...", true, true); else if (mProDialog.isShowing() || (latestOrFailDialog != null && latestOrFailDialog .isShowing())) return; } final Handler handler = new Handler() { public void handleMessage(Message msg) { // 进度条对话框不显示 - 检测结果也不显示 if (mProDialog != null && !mProDialog.isShowing()) { return; } // 关闭并释放释放进度条对话框 if (isShowMsg && mProDialog != null) { mProDialog.dismiss(); mProDialog = null; } // 显示检测结果 if (msg.what == 1) { mUpdate = (Update) msg.obj; if (mUpdate != null) { if (curVersionCode < mUpdate.getVersionCode()) { apkUrl = mUpdate.getDownloadUrl(); updateMsg = mUpdate.getUpdateLog(); showNoticeDialog(); } else if (isShowMsg) { showLatestOrFailDialog(DIALOG_TYPE_LATEST); } } } else if (isShowMsg) { showLatestOrFailDialog(DIALOG_TYPE_FAIL); } } }; new Thread() { public void run() { Message msg = new Message(); try { Update update = ApiClient .checkVersion((AppContext) mContext .getApplicationContext()); msg.what = 1; msg.obj = update; } catch (Exception e) { e.printStackTrace(); } handler.sendMessage(msg); } }.start(); }
/** * 显示版本更新通知对话框 */ private void showNoticeDialog() { AlertDialog.Builder builder = new Builder(mContext); builder.setTitle("软件版本更新"); builder.setMessage(updateMsg); builder.setPositiveButton("立即更新", new OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { dialog.dismiss(); showDownloadDialog(); } }); builder.setNegativeButton("以后再说", new OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { dialog.dismiss(); } }); noticeDialog = builder.create(); noticeDialog.show(); }
/** * 显示下载对话框 */ private void showDownloadDialog() { AlertDialog.Builder builder = new Builder(mContext); builder.setTitle("正在下载新版本");
final LayoutInflater inflater = LayoutInflater.from(mContext); View v = inflater.inflate(R.layout.update_progress, null); mProgress = (ProgressBar) v.findViewById(R.id.update_progress); mProgressText = (TextView) v.findViewById(R.id.update_progress_text);
builder.setView(v); builder.setNegativeButton("取消", new OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { dialog.dismiss(); interceptFlag = true; } }); builder.setOnCancelListener(new OnCancelListener() { @Override public void onCancel(DialogInterface dialog) { dialog.dismiss(); interceptFlag = true; } }); downloadDialog = builder.create(); downloadDialog.setCanceledOnTouchOutside(false); downloadDialog.show();
downloadApk(); }
/** * 显示'已经是最新'或者'无法获取版本信息'对话框 */ private void showLatestOrFailDialog(int dialogType) { if (latestOrFailDialog != null) { // 关闭并释放之前的对话框 latestOrFailDialog.dismiss(); latestOrFailDialog = null; } AlertDialog.Builder builder = new Builder(mContext); builder.setTitle("系统提示"); if (dialogType == DIALOG_TYPE_LATEST) { builder.setMessage("您当前已经是最新版本"); } else if (dialogType == DIALOG_TYPE_FAIL) { builder.setMessage("无法获取版本更新信息"); } builder.setPositiveButton("确定", null); latestOrFailDialog = builder.create(); latestOrFailDialog.show(); }
private Runnable mdownApkRunnable = new Runnable() { @Override public void run() { try { String apkName = "OSChinaApp_" + mUpdate.getVersionName() + ".apk"; String tmpApk = "OSChinaApp_" + mUpdate.getVersionName() + ".tmp"; // 判断是否挂载了SD卡 String storageState = Environment.getExternalStorageState(); if (storageState.equals(Environment.MEDIA_MOUNTED)) { savePath = Environment.getExternalStorageDirectory() .getAbsolutePath() + "/OSChina/Update/"; File file = new File(savePath); if (!file.exists()) { file.mkdirs(); } apkFilePath = savePath + apkName; tmpFilePath = savePath + tmpApk; }
// 没有挂载SD卡,无法下载文件 if (apkFilePath == null || apkFilePath == "") { mHandler.sendEmptyMessage(DOWN_NOSDCARD); return; }
File ApkFile = new File(apkFilePath);
// 是否已下载更新文件 if (ApkFile.exists()) { downloadDialog.dismiss(); installApk(); return; }
// 输出临时下载文件 File tmpFile = new File(tmpFilePath); FileOutputStream fos = new FileOutputStream(tmpFile);
URL url = new URL(apkUrl); HttpURLConnection conn = (HttpURLConnection) url .openConnection(); conn.connect(); int length = conn.getContentLength(); InputStream is = conn.getInputStream();
// 显示文件大小格式:2个小数点显示 DecimalFormat df = new DecimalFormat("0.00"); // 进度条下面显示的总文件大小 apkFileSize = df.format((float) length / 1024 / 1024) + "MB";
int count = 0; byte buf[] = new byte[1024];
do { int numread = is.read(buf); count += numread; // 进度条下面显示的当前下载文件大小 tmpFileSize = df.format((float) count / 1024 / 1024) + "MB"; // 当前进度值 progress = (int) (((float) count / length) * 100); // 更新进度 mHandler.sendEmptyMessage(DOWN_UPDATE); if (numread <= 0) { // 下载完成 - 将临时下载文件转成APK文件 if (tmpFile.renameTo(ApkFile)) { // 通知安装 mHandler.sendEmptyMessage(DOWN_OVER); } break; } fos.write(buf, 0, numread); } while (!interceptFlag);// 点击取消就停止下载
fos.close(); is.close(); } catch (MalformedURLException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } };
/** * 下载apk * * @param url */ private void downloadApk() { downLoadThread = new Thread(mdownApkRunnable); downLoadThread.start(); }
/** * 安装apk * * @param url */ private void installApk() { File apkfile = new File(apkFilePath); if (!apkfile.exists()) { return; } Intent i = new Intent(Intent.ACTION_VIEW); i.setDataAndType(Uri.parse("file://" + apkfile.toString()), "application/vnd.android.package-archive"); mContext.startActivity(i); } }
3:应用程序更新实体类
package lcl.android.core;
import java.io.IOException; import java.io.InputStream; import java.io.Serializable;
import lcl.android.utility.StringUtils;
import org.xmlpull.v1.XmlPullParser; import org.xmlpull.v1.XmlPullParserException;
import android.util.Xml;
/** * 应用程序更新实体类 * * @author liux (http://my.oschina.net/liux) * @version 1.0 * @created 2012-3-21 */ public class Update implements Serializable {
private static final long serialVersionUID = -1356876892850891288L; public final static String UTF8 = "UTF-8"; public final static String NODE_ROOT = "oschina";
private int versionCode; private String versionName; private String downloadUrl; private String updateLog;
public int getVersionCode() { return versionCode; }
public void setVersionCode(int versionCode) { this.versionCode = versionCode; }
public String getVersionName() { return versionName; }
public void setVersionName(String versionName) { this.versionName = versionName; }
public String getDownloadUrl() { return downloadUrl; }
public void setDownloadUrl(String downloadUrl) { this.downloadUrl = downloadUrl; }
public String getUpdateLog() { return updateLog; }
public void setUpdateLog(String updateLog) { this.updateLog = updateLog; }
public static Update parse(InputStream inputStream) throws IOException, Exception { Update update = null; // 获得XmlPullParser解析器 XmlPullParser xmlParser = Xml.newPullParser(); try { xmlParser.setInput(inputStream, UTF8); // 获得解析到的事件类别,这里有开始文档,结束文档,开始标签,结束标签,文本等等事件。 int evtType = xmlParser.getEventType(); // 一直循环,直到文档结束 while (evtType != XmlPullParser.END_DOCUMENT) { String tag = xmlParser.getName(); switch (evtType) { case XmlPullParser.START_TAG: // 通知信息 if (tag.equalsIgnoreCase("android")) { update = new Update(); } else if (update != null) { if (tag.equalsIgnoreCase("versionCode")) { update.setVersionCode(StringUtils.toInt( xmlParser.nextText(), 0)); } else if (tag.equalsIgnoreCase("versionName")) { update.setVersionName(xmlParser.nextText()); } else if (tag.equalsIgnoreCase("downloadUrl")) { update.setDownloadUrl(xmlParser.nextText()); } else if (tag.equalsIgnoreCase("updateLog")) { update.setUpdateLog(xmlParser.nextText()); } } break; case XmlPullParser.END_TAG: break; } // 如果xml没有结束,则导航到下一个节点 evtType = xmlParser.next(); } } catch (XmlPullParserException e) {
} finally { inputStream.close(); } return update; } }
4:获取网络版本信息
/** * API客户端接口:用于访问网络数据 * * @author liux (http://my.oschina.net/liux) * @version 1.0 * @created 2012-3-21 */ public class ApiClient { /** * 检查版本更新 * * @param url * @return */ public static Update checkVersion(AppContext appContext) throws Exception { try { return Update.parse(HtmlRegexpUtil .GetInputStreamByUrl(“http://files.cnblogs.com/luomingui/MobileAppVersion.xml”)); } catch (Exception e) { throw e; } } } 5:发布到网络
image
android 在线升级借助开源中国App源码的更多相关文章
- Android 轻量级ORM数据库开源框架ActiveAndroid 源码分析
ActiveAndroid 项目地址在https://github.com/pardom/ActiveAndroid 关于他的详细介绍和使用步骤 可以看下面两篇文章: https://github.c ...
- Android之开源中国客户端源码分析(一)
程序启动第一个界面类: net.oschina.app.AppStart功能描述:一张图片代码细节描述:一个透明度的动画效果,效果动画完成后自动启动新的Activity(Main) 基本BaseAct ...
- Android之开源中国客户端源码分析(二)
1. 加载动画圈实现 <ProgressBar android:id="@+id/main_head_progress" style="@style/loading ...
- android动画源码合集、动态主题框架、社交app源码等
Android精选源码 仿MIUI果冻视图-BouncingJellyView 一个快速易用的动态主题框架 android动画效果集合源码 android使用Kotlin开发的Dribbb ...
- 优化Recorder H5录音:可边录边转码上传服务器,支持微信提供Android IOS Hybrid App源码
Recorder H5 GitHub开源库随着支持功能的增多,音频转码处理效率渐渐的跟不上需求了,近期抽时间对音频转码部分进行了升级优化,以支持更多实用的功能. 另外IOS的Hybrid App也完成 ...
- 140款Android开源优秀项目源码
BeautifulRefreshLayout-漂亮的美食下拉刷新 https://github.com/android-cjj/BeautifulRefreshLayout/tree/Beautifu ...
- 【原】Android热更新开源项目Tinker源码解析系列之一:Dex热更新
[原]Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Tinker是微信的第一个开源项目,主要用于安卓应用bug的热修复和功能的迭代. Tinker github地址:http ...
- 【原】Android热更新开源项目Tinker源码解析系列之二:资源文件热更新
上一篇文章介绍了Dex文件的热更新流程,本文将会分析Tinker中对资源文件的热更新流程. 同Dex,资源文件的热更新同样包括三个部分:资源补丁生成,资源补丁合成及资源补丁加载. 本系列将从以下三个方 ...
- 基于Android开发的天气预报app(源码下载)
原文:基于Android开发的天气预报app(源码下载) 基于AndroidStudio环境开发的天气app -系统总体介绍:本天气app使用AndroidStudio这个IDE工具在Windows1 ...
随机推荐
- 超强语感训练文章(Provided by Rocky teacher Prince)
Content: Class1 My name is Prince Class2 Welcome to our hotel Class3 We’re not afraid of problems Cl ...
- cobbler工作流分析
官网 http://cobbler.github.io/ 介绍 Cobbler是一个快速网络安装linux的服务,而且在经过调整也可以支持网络安装windows.该工具使用python开发,小巧轻便, ...
- CSS3-实现单选框radio的小动画
在微信上看到一个教程文,觉得制作的小动画还是很有意思的,自己也试验了一下.一开始动画怎么都不执行(我用的HB),因为内置浏览器对css3的不兼容.加上各种浏览器前缀后就好了.但是旋转那个效果,在HB里 ...
- SQL ALTER TABLE 语句在项目中的使用
1.在实际的项目开发过程中,之前已经创建好的实体类可能需要增加/删除字段,亦或是更改已有字段的属性,比如主键的增长策略从自增型改为UUID型,那么就会涉及到 SQL 中 alter table 语句的 ...
- SQL Server编程(06)触发器
SQL Server 通过触发器用来保证业务逻辑和数据的完整性.在SQL Server中,触发器是一种特殊类型的存储过程,可在执行语言事件时自动触发.SQL Server中触发器包括三种:DML触发器 ...
- centos linux安装telnet 过程及问题(源于内部tomcat网站,外部无法访问)
首先本地没有telnet客户端及服务器 root权限下安装 yum install telnet yum install telnet-server vi /etc/xinetd.d/telnet 这 ...
- 《css3实战》读书笔记 第一章 基于CSS需求而编写的HTML.
笔记说明 <CSS3实战手册第3版(影印版)>可以消除Web设计工作的痛苦,并且带给你:HTML--重新入门.如果你是HTML新手,你会学到如何以CSS友好的方式进行基本页面构造.若你是H ...
- push submodule
git status git add sparx git commit -m "message" git push
- OSX10.11 删除系统自带的软件
之前一直用sudo rm - rf 系统的浏览器名字 正常删除safari 升级到10.11后,完全没作用了 需要关闭系统的什么安全模式 csrutil disable 再进入系统使用此命令可正常删除 ...
- Web 安全测试
http://blog.sina.com.cn/s/blog_a1bbddc70101dt12.html http://blog.csdn.net/pdn2000/article/details/64 ...