详解Android app自动更新总结(已适配9.0)

1.配置:

1.1 AndroidManifest.xml中添加权限和FileProvider:

 <uses-permission android:name="android.permission.INTERNET"/>
 <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
 <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
 <uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES" />
  <provider
   android:name="androidx.core.content.FileProvider"
   android:authorities="com.fengzhi.wuyemanagement.fileprovider"
   android:grantUriPermissions="true"
   android:exported="false">
   <meta-data
    android:name="android.support.FILE_PROVIDER_PATHS"
    android:resource="@xml/file_paths" />
  </provider>

1.2 新建文件(路径:res\xml\file_paths.xml):

<paths>
 <external-path path="." name="external_storage_root" />
</paths>

1.3 (app的)build.gradle:

 implementation "com.lzy.net:okgo:3.0.4"//okgo 网络请求
 implementation 'com.google.code.gson:gson:2.8.2'//gson
 implementation "org.permissionsdispatcher:permissionsdispatcher:4.3.1"//权限
 annotationProcessor "org.permissionsdispatcher:permissionsdispatcher-processor:4.3.1"//权限

2.这里以点击按钮进行更新为例:

 2.1 核心代码:

 private int version;
 /* 更新进度条 */
 private ProgressBar mProgress;
 private AlertDialog mDownloadDialog;

--------------------------------------------------------------------------------------------------------------------

 //点击按钮,检查权限,,,检查更新的方法
 @NeedsPermission({Manifest.permission.READ_EXTERNAL_STORAGE,
   Manifest.permission.WRITE_EXTERNAL_STORAGE,
   Manifest.permission.REQUEST_INSTALL_PACKAGES})
 protected void checkUpdate() {
  showLoadingDialog("检测更新中...");
  version = AppUpdateUtil.getAppVersionCode(this);//检查当前版本号
//  调用方法,,,接口的具体实现,接收传过来的参数,再调自己的方法,
  requestAppUpdate(version, new DataRequestListener<UpdateAppBean>() {
   @Override
   public void success(UpdateAppBean data) {
//    返回的json,getStatus为0时,去下载apk文件,这里是下载apk文件的方法
    updateApp(data.getData().getApk_url());
   }

   @Override
   public void fail(String msg) {
//    返回的json,getStatus为1时,提示:"已是最新版本!"
    SToast(msg);
    dismissLoadingDialog();
   }
  });
 }

 //检查版本号,第一次请求(post),,,UpdateAppBean根据服务器返回生成
 private void requestAppUpdate(int version, final DataRequestListener<UpdateAppBean> listener) {
  OkGo.<String>post(Const.HOST_URL + Const.UPDATEAPP).params("version", version).execute(new StringCallback() {
   @Override
   public void onSuccess(Response<String> response) {
    Gson gson = new Gson();
    UpdateAppBean updateAppBean = gson.fromJson(response.body(), UpdateAppBean.class);
    if (updateAppBean.getStatus() == 0) {
     listener.success(updateAppBean);
    } else {
     listener.fail(updateAppBean.getMsg());
    }
   }

   @Override
   public void onError(Response<String> response) {
    listener.fail("服务器连接失败");
    dismissLoadingDialog();
   }
  });
 }

 //如果有新版本,提示有新的版本,然后下载apk文件
 private void updateApp(String apk_url) {
  dismissLoadingDialog();
  DialogUtils.getInstance().showDialog(this, "发现新的版本,是否下载更新?",
    new DialogUtils.DialogListener() {
   @Override
   public void positiveButton() {
    downloadApp(apk_url);
   }
  });
 }

 //下载apk文件并跳转(第二次请求,get)
 private void downloadApp(String apk_url) {
  OkGo.<File>get(apk_url).tag(this).execute(new FileCallback() {
   @Override
   public void onSuccess(Response<File> response) {
    String filePath = response.body().getAbsolutePath();
    Intent intent = IntentUtil.getInstallAppIntent(mContext, filePath);
//    测试过这里必须用startactivity,不能用stratactivityforresult
    mContext.startActivity(intent);
    dismissLoadingDialog();
    mDownloadDialog.dismiss();
    mDownloadDialog=null;
   }

   @Override
   public void downloadProgress(Progress progress) {
//      showDownloadDialog();
//      mProgress.setProgress((int) (progress.fraction * 100));
    if (mDownloadDialog == null) {
     // 构造软件下载对话框
     AlertDialog.Builder builder = new AlertDialog.Builder(mContext);
     builder.setTitle("正在更新");
     // 给下载对话框增加进度条
     final LayoutInflater inflater = LayoutInflater.from(mContext);
     View v = inflater.inflate(R.layout.item_progress, null);
     mProgress = (ProgressBar) v.findViewById(R.id.update_progress);
     builder.setView(v);
     mDownloadDialog = builder.create();
     mDownloadDialog.setCancelable(false);
     mDownloadDialog.show();
    }
    mProgress.setProgress((int) (progress.fraction * 100));
   }
  });
 }

2.2 DataRequestListener:

public interface DataRequestListener<T> {
 //请求成功
 void success(T data);
 //请求失败
 void fail(String msg);
}

接下来是工具类,来自github,参考, https://github.com/vondear/RxTool

2.3 AppUpdateUtil:

 /**
  * 获取App版本码
  *
  * @param context 上下文
  * @return App版本码
  */
 public static int getAppVersionCode(Context context) {
  return getAppVersionCode(context, context.getPackageName());
 }

2.4 IntentUtil:

public class IntentUtil {

 /**
  * 获取安装App(支持7.0)的意图
  *
  * @param context
  * @param filePath
  * @return
  */
 public static Intent getInstallAppIntent(Context context, String filePath) {
  //apk文件的本地路径
  File apkfile = new File(filePath);
  if (!apkfile.exists()) {
   return null;
  }
  Intent intent = new Intent(Intent.ACTION_VIEW);
  Uri contentUri = FileUtil.getUriForFile(context, apkfile);
  intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
  if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
   intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
  }
  intent.setDataAndType(contentUri, "application/vnd.android.package-archive");
  return intent;
 }

2.5 FileUtil:

 /**
  * 将文件转换成uri(支持7.0)
  *
  * @param mContext
  * @param file
  * @return
  */
 public static Uri getUriForFile(Context mContext, File file) {
  Uri fileUri = null;
  if (Build.VERSION.SDK_INT >= 24) {
   fileUri = FileProvider.getUriForFile(mContext, mContext.getPackageName() + ".fileprovider", file);
  } else {
   fileUri = Uri.fromFile(file);
  }
  return fileUri;
 }

3.遇到的问题

9.0手机authorities配置出错,导致无法安装, 解决办法:

1.项目中使用了Androidx,AndroidManifest.xml的配置中就必须使用androidx的fileprovider

2.这里的authorities与FileUtil.java中的要一样,我就是字母P大写了导致错误

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • 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自动更新功能

    现在一般的android软件都是需要不断更新的,当你打开某个app的时候,如果有新的版本,它会提示你有新版本需要更新.该小程序实现的就是这个功能. 该小程序的特点是,当有更新时,会弹出一个提示框,点击确定,则在通知来创建一个进度条进行下载,点击取消,则取消更新. 以下是详细代码: 1.创建布局文件notification_item.xml,用于在通知栏生成一个进度条和下载图标. <?xml version="1.0" encoding="utf-8"?>

  • Android如何实现APP自动更新

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

  • 基于Retrofit2+RxJava2实现Android App自动更新

    本文实例为大家分享了Retrofit2 RxJava2实现Android App自动更新,具体内容如下 功能解析 自动更新可以说已经是App的标配了,很多第三方平台也都支持这个功能,最近手头上的项目需要加入这个App自动更新,考虑到项目里有用到Retrofit2和RxJava2,于是打算使用它俩自己实现这个功能. 分析App自动更新,可以分为以下三个功能点: 1.APK文件的下载 2.下载进度的实时更新显示 3.下载完成后的自动安装 其中比较难的一点是下载进度的实时更新显示,更难的是如何优雅的进

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

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

  • Android App自动更新之通知栏下载

    本文实例为大家分享了Android App自动更新通知栏下载的具体代码,供大家参考,具体内容如下 版本更新说明 这里有调用UpdateService启动服务检查下载安装包等 1. 文件下载,下完后写入到sdcard 2. 如何在通知栏上显示下载进度 3. 下载完毕自动安装 4. 如何判断是否有新版本 版本更新的主类 package com.wei.update; import java.io.IOException; import java.io.InputStream; import java

  • 详解Android app自动更新总结(已适配9.0)

    1.配置: 1.1 AndroidManifest.xml中添加权限和FileProvider: <uses-permission android:name="android.permission.INTERNET"/> <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/> <uses-permission android:name=&quo

  • 详解Android项目多服务端接口适配(超简单)

    现状 Android项目如果是多服务端接口时,一般怎么弄呢? 方法1:服务器地址放在Header中 把服务器地址放在接口Header中,然后通过拦截器来动态修改请求地址而实现的.除了默认服务器的接口,其它都要加一个Header,有点麻烦.看起来也不爽,不简洁. interface ApiHeaderCase { /************************** server A ****************************/ @Headers("host:$SERVER_HOS

  • 详解Android App中创建ViewPager组件的方法

    现在很多app一打开就是一个ViewPager,然后可以用手指滑,每滑一次就换一张图,底下还会有圈圈表示说现在滑到第几章~ 通常这些图片都是放功能简介或是使用教学之类的,我的需求很简单,就是上面提到的那样而已. 有两种做法,一种是找现有套件,查了一堆资料每个都跟我推荐ViewPagerIndicator这套,我之前也看过这套,只是看起来需要有fragment再加上google play范例好像载不到了,所以只好自己实做一个. Viewpager的实作可参考Android ViewPager使用详

  • 详解android进行异步更新UI的四种方式

    大家都知道由于性能要求,Android要求只能在UI线程中更新UI,要想在其他线程中更新UI,我大致总结了4种方式,欢迎补充纠正: 使用Handler消息传递机制: 使用AsyncTask异步任务: 使用runOnUiThread(action)方法: 使用Handler的post(Runnabel r)方法: 下面分别使用四种方式来更新一个TextView. 1.使用Handler消息传递机制 package com.example.runonuithreadtest; import andr

  • 详解Android App中的AsyncTask异步任务执行方式

    基本概念 AsyncTask:异步任务,从字面上来说,就是在我们的UI主线程运行的时候,异步的完成一些操作.AsyncTask允许我们的执行一个异步的任务在后台.我们可以将耗时的操作放在异步任务当中来执行,并随时将任务执行的结果返回给我们的UI线程来更新我们的UI控件.通过AsyncTask我们可以轻松的解决多线程之间的通信问题. 怎么来理解AsyncTask呢?通俗一点来说,AsyncTask就相当于Android给我们提供了一个多线程编程的一个框架,其介于Thread和Handler之间,我

  • 详解Android App中使用VideoView来实现视频播放的方法

    通过VideoView播放视频的步骤: 1.在界面布局文件中定义VideoView组件,或在程序中创建VideoView组件 2.调用VideoView的如下两个方法来加载指定的视频 (1)setVidePath(String path):加载path文件代表的视频 (2)setVideoURI(Uri uri):加载uri所对应的视频 3.调用VideoView的start().stop().psuse()方法来控制视频的播放 VideoView通过与MediaController类结合使用,

  • 详解Android App卸载后跳转到指定的反馈页面的方法

    很多人也许会问:360被卸载之后会跳转到指定的反馈页面,是怎么弄的? 其实这个问题的核心就在于:应用被卸载了,如果能够做到后续的代码逻辑继续执行 我们再来仔细分析一下场景和流程 一个应用被用户卸载肯定是有理由的,而开发者却未必能得知这一重要的理由,毕竟用户很少会主动反馈建议,多半就是用得不爽就卸,如果能在被卸载后获取到用户的一些反馈,那对开发者进一步改进应用是非常有利的.目前据我所知,国内的Android应用中实现这一功能的只有360手机卫士.360平板卫士,那么如何实现这一功能的? 我们可以把

  • 详解Android App中ViewPager使用PagerAdapter的方法

    PageAdapter是一个抽象类,直接继承于Object,导入包android.support.v4.view.PagerAdapter即可使用. 要使用PagerAdapter, 首先要继承PagerAdapter类,至少覆盖以下方法: 在每次创建ViewPager或滑动过程中,以下四个方法都会被调用,而instantiateItem和destroyItem中的方法要自己去实现. public abstract int getCount(); 这个方法,是获取当前窗体界面数 public a

随机推荐