Android应用App更新实例详解

前言:现在一般的Android软件都是需要不断更新的,当你打开某个app的时候,如果有新的版本,它会提示你有新版本需要更新。该项目实现的就是这个功能。并且有强制更新和更新提示两种方式,当有更新时,会弹出一个提示框,点击下载,则在通知来创建一个进度条进行下载,点击取消,则取消更新。

效果:

开发环境:AndroidStudio2.1.2+gradle-2.10

部分代码:

public class UpdateVersionController {
 private Context context;
 //更新文件的实例
 private AppUpdateInfo info;
 //当前版本号
 private int versionCode;
 //提示用户更新的dialog
 private Dialog dialog;
 //下载进度条
 private ProgressDialog pd;

 public static UpdateVersionController getInstance(Context context) {
 return new UpdateVersionController(context);
 }
 public UpdateVersionController(Context context) {
 this.context = context;
 }
 public void normalCheckUpdateInfo() {
 //获取版本号:这里的版本号在项目的build.gradle中是可以看到的,看复制过来的参数
 /**
 defaultConfig {
 applicationId "com.zhh.test"
 minSdkVersion 16
 targetSdkVersion 23
 versionCode 1
 versionName "1.0"
 }
 */
 versionCode = getVerCode(context);//等于19
 checkVersionTask();
 }
 public void forceCheckUpdateInfo() {//强制更新一般不用
 versionCode = getVerCode(context);//等于19
 info = new AppUpdateInfo();
 info.setUrl("http://60.210.21.89/appdl.hicloud.com/dl/appdl/application/apk/dd/ddeabc7e64d64e4287340ffb89c592a8/com.hellotext.1309171635.apk?sign=baidu@baidu&wshc_tag=0&wsts_tag=58c7bde2&wsid_tag=7c5a47c8&wsiphost=ipdbm");
 info.setVercode(2);//每次更新都靠它
 info.setVername("2.0");//版本名字
 info.setApkname("com.hellotext.1309171635.apk");
 info.setAppname("Hello");
 info.setForceUpp("yes");
 info.setUppcontent("1. Hello啊哟更新了\n2. 英文的,界面很好看.\n3. 界面效果优化");//更新内容
 updateApp();
 }
 /**
 * 步骤一:获取版本信息
 */
 private void checkVersionTask() {
 //网络加载获取app新版版本信息
 //这里不做请求直接赋值
 info = new AppUpdateInfo();
 info.setUrl("http://60.210.21.89/appdl.hicloud.com/dl/appdl/application/apk/dd/ddeabc7e64d64e4287340ffb89c592a8/"+
            "com.hellotext.1309171635.apk?sign=baidu@baidu&wshc_tag=0&wsts_tag=58c7bde2&wsid_tag=7c5a47c8&wsiphost=ipdbm");
 info.setVercode(2);//每次更新都靠它
 info.setVername("2.0");//版本名字
 info.setApkname("com.hellotext.1309171635.apk");
 info.setAppname("Hello");
 info.setForceUpp("no");
 info.setUppcontent("1. Hello啊哟更新了\n2. 英文的,界面很好看.\n3. 界面效果优化");//更新内容
 updateApp();
 }
 private void updateApp() {
 if (null != info && info.getVercode() > versionCode) {//20>19可更新
 showUpdataDialog();
 } else {
 Toast.makeText(context, "已经是最新版本啦~", Toast.LENGTH_SHORT).show();
 }
 }
 private Button cancelBtn;
 /**
 * 步骤二:弹出对话框提示用户更新
 */
 protected void showUpdataDialog() {
 dialog = new Dialog(context, android.R.style.Theme_Dialog);
 dialog.getWindow().setBackgroundDrawableResource(android.R.color.transparent);
 dialog.setContentView(R.layout.activity_updater);
 dialog.setCanceledOnTouchOutside(false);
 dialog.setCancelable(false);
 ((TextView) dialog.findViewById(R.id.content)).setText(info.getUppcontent());
 cancelBtn = (Button) dialog.findViewById(R.id.cancel);
 cancelBtn.setVisibility("yes".equals(info.getForceUpp()) ? View.GONE : View.VISIBLE);
 // 取消更新
 cancelBtn.setOnClickListener(new View.OnClickListener() {
 @Override
 public void onClick(View v) {
 dialog.dismiss();
 }
 });
 // 确认更新
 dialog.findViewById(R.id.ok).setOnClickListener(new View.OnClickListener() {
 @Override
 public void onClick(View v) {
 dialog.dismiss();
 downLoadApk();
 }
 });
 dialog.findViewById(R.id.market).setOnClickListener(new View.OnClickListener() {
 @Override
 public void onClick(View v) {
 Intent intent = new Intent(Intent.ACTION_VIEW);
 intent.setData(Uri.parse("market://details?id=Hello"));
 context.startActivity(intent);
 }
 });
 dialog.show();
 }
 /**
 * 步骤三:下载文件
 */
 private void downLoadApk() {
 // 进度条对话框
 pd = new ProgressDialog(context);
 pd.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
 pd.setMessage("下载中...");
 pd.setCanceledOnTouchOutside(false);
 pd.setCancelable(false);
 // 监听返回键--防止下载的时候点击返回
 pd.setOnKeyListener(new OnKeyListener() {
 @Override
 public boolean onKey(DialogInterface dialog, int keyCode, KeyEvent event) {
 if (keyCode == KeyEvent.KEYCODE_BACK && event.getRepeatCount() == 0) {
 Toast.makeText(context, "正在下载请稍后", Toast.LENGTH_SHORT).show();
 return true;
 } else {
 return false;
 }
 }
 });
 // Sdcard不可用
 if (!Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {
 Toast.makeText(context, "SD卡不可用~", Toast.LENGTH_SHORT).show();
 } else {
 pd.show();
 //下载的子线程
 new Thread() {
 @Override
 public void run() {
 try {
 // 在子线程中下载APK文件
 File file = getFileFromServer(info.getUrl(), pd);
 sleep(1000);
 // 安装APK文件
 installApk(file);
 pd.dismiss(); // 结束掉进度条对话框
 } catch (Exception e) {
 Toast.makeText(context, "文件下载失败了~", Toast.LENGTH_SHORT).show();
 pd.dismiss();
 e.printStackTrace();
 }
 }
 }.start();
 }
 }
 /**
 * 从服务器下载apk
 */
 public File getFileFromServer(String path, ProgressDialog pd) throws Exception {
 // 如果相等的话表示当前的sdcard挂载在手机上并且是可用的
 if (Environment.getExternalStorageState().equals(
 Environment.MEDIA_MOUNTED)) {
 URL url = new URL(path);
 HttpURLConnection conn = (HttpURLConnection) url.openConnection();
 conn.setConnectTimeout(5000);
 // 获取到文件的大小
 pd.setMax(conn.getContentLength() / 1024);
 InputStream is = conn.getInputStream();

 File file = new File(Environment.getExternalStorageDirectory().getPath()
 + "/blibao/merchant", "i_blibao_shop.apk");
 //判断文件夹是否被创建
 if (!file.getParentFile().exists()) {
 file.getParentFile().mkdirs();
 }
 FileOutputStream fos = new FileOutputStream(file);
 BufferedInputStream bis = new BufferedInputStream(is);
 byte[] buffer = new byte[1024];
 int len;
 int total = 0;
 while ((len = bis.read(buffer)) != -1) {
 fos.write(buffer, 0, len);
 total += len;
 // 获取当前下载量
 pd.setProgress(total / 1024);
 }
 fos.close();
 bis.close();
 is.close();
 return file;
 } else {
 return null;
 }
 }
 /**
 * 安装apk
 */
 protected void installApk(File file) {
 Intent intent = new Intent();
 // 执行动作
 intent.setAction(Intent.ACTION_VIEW);
 // 执行的数据类型
 intent.setDataAndType(Uri.fromFile(file), "application/vnd.android.package-archive");
 intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
 context.startActivity(intent);
 }
 /**
 * 获取版本名
 */
 public static String getVerName(Context context) {
 String verName = "";
 try {
 // 获取packagemanager的实例
 PackageManager packageManager = context.getPackageManager();
 // getPackageName()是你当前类的包名,0代表是获取版本信息
 PackageInfo packInfo = packageManager.getPackageInfo(
 context.getPackageName(), 0);

 verName = packInfo.versionName;
 } catch (Exception e) {
 e.printStackTrace();
 }
 return verName;
 }
 /**
 * 获取版本号
 */
 public static int getVerCode(Context context) {
 int verCode = -1;
 try {
 // 获取packagemanager的实例
 PackageManager packageManager = context.getPackageManager();
 // getPackageName()是你当前类的包名,0代表是获取版本信息
 PackageInfo packInfo = packageManager.getPackageInfo(
 context.getPackageName(), 0);

 verCode = packInfo.versionCode;
 } catch (Exception e) {
 e.printStackTrace();
 }
 return verCode;
 }
}

源码下载

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持我们!

(0)

相关推荐

  • android实现通知栏下载更新app示例

    1.设计思路,使用VersionCode定义为版本升级参数.android为我们定义版本提供了2个属性: 复制代码 代码如下: <manifest package="com.cnblogs.tianxia.subway"android:versionCode="1" <!--Integer类型,系统不显示给用户-->android:versionName="1.0"<!--String类型,系统显示用户-->>

  • Android App实现应用内部自动更新的最基本方法示例

    这只是初步的实现,并没有加入自动编译等功能.需要手动更改更新的xml文件和最新的apk.    共涉及到四个文件! 一.客户端 AndroidUpdateTestActivity:程序首页 main.xml:首页布局 Update:更新类 softupdate_progress:更新等待界面 Updage package majier.test; import java.io.File; import java.io.FileOutputStream; import java.io.IOExce

  • Android-App增量更新的使用姿势

    简述 增量更新,根据字面理解,就是下载增加的那部分来达到更新的目的,实际就是这个意思. 原理 用一个旧的Apk安装与一个新的Apk安装包使用 bsdiff工具 ,执行命令生成一个差异文件,此差异文件就是我们修改需要更新下载的那部分. 引入代码及so文件 首先,根据你的系统的架构选择不同的so文件放到你的工程中 接着,需要把加载so文件的Java类引入到你的工程中,引入时,需注意,不能修改这个类的包名. 到此,增量更新引入完成. 使用 下载bsdiff工具,然后执行命令:bsdiff,会显示出命令

  • Android App增量更新详解及实例代码

    Android App增量更新实例--Smart App Updates        介绍 你所看到的,是一个用于Android应用程序增量更新的开源库. 包括客户端.服务端两部分代码. 原理 自从 Android 4.1 开始,Google引入了应用程序的增量更新. Link: http://developer.android.com/about/versions/jelly-bean.html Smart app updates is a new feature of Google Pla

  • Android应用强制更新APP的示例代码

    Android应用强制更新的用途十分广泛,特别上刚上线的应用肯定会存在或多或少的bug,特别是涉及移动支付这一块的内容,如果出错了会造成比较大的损失,所以强制更新显得尤为重要. 一般来说,强制更新的策略就是: 应用启动时请求后台,后台发送应用最新版本的信息(包括应用版本号.名称.更新内容说明.下载包的服务器地址.是否强制更新的标志位)等等. 下面我们就将根据以上思路来写实现代码. 1.AndroidManifest配置版本信息 在AndroidManifest.xml里定义了每个Android

  • Android如何实现APP自动更新

    先来看看要实现的效果图: 对于安卓用户来说,手机应用市场说满天飞可是一点都不夸张,比如小米,魅族,百度,360,机锋,应用宝等等,当我们想上线一款新版本APP时,先不说渠道打包的麻烦,单纯指上传APP到各大应用市场的工作量就已经很大了,好不容易我们把APP都上传完了,突然发现一个会导致应用闪退的小Bug,这时那个崩溃啊,明明不是很大的改动,难道我们还要再去重新去把各大应用市场的版本再上传更新一次?相信我,运营人员肯定会弄死你的!! 有问题,自然就会有解决问题的方案,因此我们就会想到如果在APP里

  • Android应用APP自动更新功能的代码实现

    由于Android项目开源所致,市面上出现了N多安卓软件市场.为了让我们开发的软件有更多的用户使用,我们需要向N多市场发布,软件升级后,我们也必须到安卓市场上进行更新,给我们增加了工作量.因此我们有必要给我们的Android应用增加自动更新的功能. 既然实现自动更新,我们首先必须让我们的应用知道是否存在新版本的软件,因此我们可以在自己的网站上放置配置文件,存放软件的版本信息: <update> <version>2</version> <name>baidu

  • Android应用App更新实例详解

    前言:现在一般的Android软件都是需要不断更新的,当你打开某个app的时候,如果有新的版本,它会提示你有新版本需要更新.该项目实现的就是这个功能.并且有强制更新和更新提示两种方式,当有更新时,会弹出一个提示框,点击下载,则在通知来创建一个进度条进行下载,点击取消,则取消更新. 效果: 开发环境:AndroidStudio2.1.2+gradle-2.10 部分代码: public class UpdateVersionController { private Context context;

  • Android 中ViewPager重排序与更新实例详解

    Android 中ViewPager重排序与更新实例详解 最近的项目中有栏目订阅功能,在更改栏目顺序以后需要更新ViewPager.类似于网易新闻的频道管理. 在重新排序之后调用了PagerAdapter的notifyDataSetChanged方法,发现ViewPager并没有更新,于是我开始跟踪源码,在调用PagerAdapter的notifyDataSetChanged方法后,会触发Viewpager的dataSetChanged方法. void dataSetChanged() { //

  • Android的搜索框架实例详解

    基础知识 Android的搜索框架将代您管理的搜索对话框,您不需要自己去开发一个搜索框,不需要担心要把搜索框放什么位置,也不需要担心搜索框影响您当前的界面.所有的这些工作都由SearchManager类来为您处理(以下简称"搜索管理器"),它管理的Android搜索对话框的整个生命周期,并执行您的应用程序将发送的搜索请求,返回相应的搜索关键字. 当用户执行一个搜索,搜索管理器将使用一个专门的Intent把搜索查询的关键字传给您在配置文件中配置的处理搜索结果的Activity.从本质上讲

  • Android 中 ActivityLifecycleCallbacks的实例详解

    Android 中 ActivityLifecycleCallbacks的实例详解           以上就是使用ActivityLifecycleCallbacks的实例,代码注释写的很清楚大家可以参考下, MyApplication如下: package com.cc; import java.util.LinkedList; import android.app.Activity; import android.app.Application; import android.os.Bun

  • Android 自定义标题栏的实例详解

     Android 自定义标题栏的实例详解 开发 Android APP 经常会用到自定义标题栏,而有多级页面的情况下还需要给自定义标题栏传递数据. 本文要点: 自定义标题填充不完整 自定义标题栏返回按钮的点击事件 一.代码 这里先介绍一下流程: 1. 创建一个标题栏布局文件 mytitlebar.xml 2. 在style.xml中创建 mytitlestyle 主题 3. 创建类 CustomTitleBar 4. 在需要自定义标题栏的Activity的OnCreate方法中实例化 Custo

  • Android 上下滚动TextSwitcher实例详解

    Android 上下滚动TextSwitcher实例详解 1.在activity中需要代码声明 textSwitcher = (TextSwitcher)findViewById(R.id.text_switcher); textSwitcher.setFactory(new ViewFactory() { @Override public View makeView() { TextView tv = new TextView(MainActivity.this); tv.setTextSiz

  • Android原生音量控制实例详解

    本文主要涉及AudioService.还是基于5.1.1版本的代码. AudioService.java文件位于/framework/base/media/java/android/media/下. 音量控制是AudioService最重要的功能之一.先总结一下: AudioService音量管理的核心是VolumeStreamState.它保存了一个流类型所有的音量信息. VolumeStreamState保存了运行时的音量信息,而音量的生效则是在底层AudioFlinger完成的.所以进行音

  • Android 帧动画的实例详解

    Android 帧动画的实例详解 对于 Android 帧动画 大体上可以理解成 一张张图片 按一定顺序切换, 这样当连续几张图是一组动画时,就可以连起来了看成是一个小电影,你懂得 好得,比就装到这里,下面开始进入正题,由于产品需求 需要做一个 声音喇叭动态切换的样式,我特么第一就想到是帧动画切换,然后就百度了一些资料,发现 真的, 现在这个网上太多的资料是 copy粘贴过来的, 一错全错,对于这种情况我只想说,made,一群垃圾, 所以今天我将带你们走进Android 正确帧动画地址. 第一步

  • Android Intent封装的实例详解

    Android Intent封装的实例详解 什么是Intent: Intent是协调应用间.组件之间的通讯和交互.通过Intent你可以启动Activity.Service.Broadcasts.更可以跨程序调用第三方组件.例如:启动拨打电话界面.音乐播放等. 组件     启动 Activity startActicity() Service startService(),bindService( ) Broadcasts sendBroadcast() 使用Intent: 栗子:在一个Act

  • Android 完全退出的实例详解

    Android 完全退出的实例详解 首先,在基类BaseActivity里,注册RxBus监听: public class BaseActivity extends AppCompatActivity { Subscription mSubscription; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); Utils.intiSySBar(thi

随机推荐