Android7.0自动更新适配 包解析异常

在Android7.0的手机上,自动更新的时候出现包解析异常,在其他的手机上没有这个问题。

原因:

Android7.0引入私有目录被限制访问和StrictMode API 。私有目录被限制访问是指在Android7.0中为了提高应用的安全性,在7.0上应用私有目录将被限制访问。StrictMode API是指禁止向你的应用外公开 file:// URI。 如果一项包含文件 file:// URI类型 的 Intent 离开你的应用,则会报出异常。

解决办法:

第一步:在AndroidManifest.xml中注册provider,provider可以向应用外提供数据。

<provider
  android:authorities="包名.fileprovider"
  android:name="android.support.v4.content.FileProvider"
  android:grantUriPermissions="true"//这是设置uri的权限
  android:exported="false">
  <meta-data
    android:name="android.support.FILE_PROVIDER_PATHS"
    android:resource="@xml/file_paths"/>//在第二步的时候会有介绍
</provider>

第二步:在res/xml中创建file_paths.xml文件。

<?xml version="1.0" encoding="utf-8"?>
<resources>
  <paths>
    <external-path path="" name="download" />
  </paths>
</resources>

第三步:贴出我的自动更新下载的代码

public class UpdateManager {
  private Context mContext;

  private static String savePath ;
  private String saveFileName ;
  private ProgressBar mProgress; //下载进度条控件
  private static final int DOWNLOADING = 1; //表示正在下载
  private static final int DOWNLOADED = 2; //下载完毕
  private static final int DOWNLOAD_FAILED = 3; //下载失败
  private int progress; //下载进度
  private boolean cancelFlag = false; //取消下载标志位

  private String serverVersion; //从服务器获取的版本号
  private String apkUrl;
//  private String apkUrl = "http://liuliu.lejuhuyu.com/AndroidApk/liuliu-dashou-app-1.0.2.apk";
  private String clientVersion; //客户端当前的版本号
  private String updateDescription = "请更新当前最新版本"; //更新内容描述信息
  private String forceUpdate; //是否强制更新
  private String update;
  private VersionBean mVersionBean;

  private AlertDialog alertDialog1, alertDialog2; //表示提示对话框、进度条对话框
  public UpdateManager(Context context,VersionBean versionBean) {
    this.mContext = context;
    this.mVersionBean = versionBean;
    apkUrl = "http://liuliu.lejuhuyu.com/AndroidApk/liuliu-dashou-app-"+versionBean.getLastVersion()+".apk";
    savePath = Environment.DIRECTORY_DOWNLOADS;
    saveFileName = savePath + "/liuliu-dashou-app-"+versionBean.getLastVersion()+".apk";
  }
  /** 显示更新对话框 */
  public void showNoticeDialog() {
    serverVersion = mVersionBean.getLastVersion();
    clientVersion = mVersionBean.getVersion();
    L.e("apkUrl="+apkUrl);
    L.e("savePath="+savePath);
    L.e("saveFileName="+saveFileName);
//    forceUpdate = StringUtils.getVersion();
//    forceUpdate = "1";
    forceUpdate = mVersionBean.getImportant();
    update = mVersionBean.getUpdate();
    //如果版本最新,则不需要更新
    if (serverVersion.equals(clientVersion))
      return;
    if (update.equals("2"))
      return;
    AlertDialog.Builder dialog = new AlertDialog.Builder(mContext);
    dialog.setTitle("发现新版本 :" + serverVersion);
    dialog.setMessage(updateDescription);
    dialog.setPositiveButton("现在更新", new DialogInterface.OnClickListener() {
      @Override
      public void onClick(DialogInterface arg0, int arg1) {
        // TODO Auto-generated method stub
        arg0.dismiss();
        showDownloadDialog();
      }
    });
    //是否强制更新
    if (forceUpdate.equals("2")) {
      dialog.setNegativeButton("待会更新", new DialogInterface.OnClickListener() {
        @Override
        public void onClick(DialogInterface arg0, int arg1) {
          // TODO Auto-generated method stub
          arg0.dismiss();
        }
      });
    }
    alertDialog1 = dialog.create();
    alertDialog1.setCancelable(false);
    alertDialog1.show();
  }
  /** 显示进度条对话框 */
  public void showDownloadDialog() {
    AlertDialog.Builder dialog = new AlertDialog.Builder(mContext);
    dialog.setTitle("正在更新");
    final LayoutInflater inflater = LayoutInflater.from(mContext);
    View v = inflater.inflate(R.layout.softupdate_progress, null);
    mProgress = (ProgressBar) v.findViewById(R.id.update_progress);
    dialog.setView(v);
    //如果是强制更新,则不显示取消按钮
//    if (forceUpdate.equals("1")) {
//      dialog.setNegativeButton("取消", new DialogInterface.OnClickListener() {
//        @Override
//        public void onClick(DialogInterface arg0, int arg1) {
//          // TODO Auto-generated method stub
//          arg0.dismiss();
//          cancelFlag = false;
//        }
//      });
//    }
    alertDialog2 = dialog.create();
    alertDialog2.setCancelable(false);
    alertDialog2.show();

    //下载apk
    downloadAPK();
  }
  DownloadManager manager;
  Cursor cursor;
  DownloadManager.Request down;
  DownloadManager.Query query;
  ContentObserver contentObserver;
  /** 下载apk的线程 */
  public void downloadAPK() {
    manager = (DownloadManager) LiuLiuApplication.getContext().getSystemService(Context.DOWNLOAD_SERVICE);
    down = new DownloadManager.Request(Uri.parse(apkUrl));
    // 设置允许使用的网络类型,这里是移动网络和wifi都可以
    down.setAllowedNetworkTypes(DownloadManager.Request.NETWORK_MOBILE
        | DownloadManager.Request.NETWORK_WIFI);
    // 显示下载界面
    down.setVisibleInDownloadsUi(true);
    // 设置下载路径和文件名
    down.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, "liuliu-dashou-app-"+mVersionBean.getLastVersion() + ".apk");
    down.setMimeType("application/vnd.android.package-archive");
    // 设置为可被媒体扫描器找到
    down.allowScanningByMediaScanner();
    down.setAllowedOverRoaming(false);
//    down.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED);
    long id = manager.enqueue(down);
    query = new DownloadManager.Query().setFilterById(id);
    contentObserver = new ContentObserver(mHandler) {
      @Override
      public void onChange(boolean selfChange) {
//        super.onChange(selfChange);
        boolean downloading = true;
        while(downloading){
          cursor = manager.query(query);
          try {
            if (cursor != null && cursor.moveToFirst()) {
              int bytes_downloaded = cursor.getInt(cursor.getColumnIndex(DownloadManager.COLUMN_BYTES_DOWNLOADED_SO_FAR));
              int bytes_total = cursor.getInt(cursor.getColumnIndex(DownloadManager.COLUMN_TOTAL_SIZE_BYTES));
              progress = (int) ((bytes_downloaded * 100) / bytes_total);
              mHandler.sendEmptyMessage(DOWNLOADING);
              if (cursor.getInt(cursor.getColumnIndex(DownloadManager.COLUMN_STATUS))==DownloadManager.STATUS_SUCCESSFUL) {
                mHandler.sendEmptyMessage(DOWNLOADED);
              }else if (cursor.getInt(cursor.getColumnIndex(DownloadManager.COLUMN_STATUS))==DownloadManager.STATUS_FAILED){
                mHandler.sendEmptyMessage(DOWNLOAD_FAILED);
              }
            }
          }catch (Exception e){
            e.printStackTrace();
            mHandler.sendEmptyMessage(DOWNLOAD_FAILED);
          }finally {
            if (cursor != null){
              downloading = false;
              cursor.close();
            }
          }
        }
      }
    };
    mContext.getContentResolver().registerContentObserver(Uri.parse("content://downloads/"),true,contentObserver);
  }

  /** 更新UI的handler */
  private Handler mHandler = new Handler() {
    @Override
    public void handleMessage(Message msg) {
      // TODO Auto-generated method stub
      switch (msg.what) {
        case DOWNLOADING:
          mProgress.setProgress(progress);
          break;
        case DOWNLOADED:
          if (alertDialog2 != null)
            alertDialog2.dismiss();
          installAPK();
          break;
        case DOWNLOAD_FAILED:
          ToastUtil.getInstance(mContext,"网络断开,请稍候再试",false).show();
          break;
        default:
          break;
      }
    }
  };

  /** 下载完成后自动安装apk */
  public void installAPK() {
    File apkFile = new File(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS),"liuliu-dashou-app-"+mVersionBean.getLastVersion() + ".apk");

    if (!apkFile.exists()) {
      return;
    }
    if (Build.VERSION.SDK_INT>=24){

      Uri apkUri = FileProvider.getUriForFile(mContext, LiuLiuApplication.getContext().getPackageName()+".fileprovider", apkFile);
      Intent install = new Intent(Intent.ACTION_VIEW);
      install.addCategory(Intent.CATEGORY_DEFAULT);
      install.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
      install.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
      install.setDataAndType(apkUri, "application/vnd.android.package-archive");
      mContext.startActivity(install);
    } else {
      Intent intent = new Intent();
      intent.setAction(Intent.ACTION_VIEW);
      intent.addCategory(Intent.CATEGORY_DEFAULT);
      intent.setType("application/vnd.android.package-archive");
      intent.setData(Uri.fromFile(apkFile));
      intent.setDataAndType(Uri.fromFile(apkFile), "application/vnd.android.package-archive");
      intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
      mContext.startActivity(intent);
    }

  }
}

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

(0)

相关推荐

  • AndroidStudio利用android-support-multidex解决64k的各种异常

    64k的各种异常 当你的应用程序和库引用达到一定规模,你遇到构建错误显示你的应用已经达到了一个Android应用程序构建体系结构的限制.早期版本的构建系统报告这个错误如下: Conversion to Dalvik format failed: Unable to execute dex: method ID not in [0, 0xffff]: 65536 或者 UNEXPECTED TOP-LEVEL EXCEPTION:  java.lang.IllegalArgumentExcepti

  • Android异常 java.lang.IllegalStateException解决方法

    Android异常详情介绍 这种异常我遇到以下两种情况: 1. java.lang.IllegalStateException: No wrapped connection. 2.java.lang.IllegalStateException: Adapter is detached. 原因: 1.单线程一次执行一个请求可以正常执行,如果使用多线程,同时执行多个请求时就会出现连接超时. 2.HttpConnection没有连接池的概念,多少次请求就会建立多少个IO,在访问量巨大的情况下服务器的I

  • Android 全局异常捕获实例详解

    Android 全局异常捕获 今天就来说说作为程序猿的我们每天都会遇到的东西bug,出bug不可怕可怕的是没有出bug时的堆栈信息,那么对于bug的信息收集就显得尤为重要了,一般用第三方bugly或者友盟等等都能轻易收集,但是由于公司不让使用第三方,而安卓正好有原生的异常收集类UncaughtExceptionHandler,那么今天博客就从这个类开始. UncaughtExceptionHandler见名知意,即他是处理我们未捕获的异常,具体使用分两步 1.实现我们自己的异常处理类 publi

  • Android跨进程抛异常的原理的实现

    今天接到了个需求,需要用到跨进程抛异常. 怎样将异常从服务端抛到客户端 也就是说在Service端抛出的异常需要可以在Client端接收.印象中binder是可以传异常的,所以aidl直接走起: // aidl文件 interface ITestExceptionAidl { boolean testThrowException(); } // service端实现 public class AidlService extends Service { @Nullable @Override pu

  • Android编程实现项目中异常捕获及对应Log日志文件保存功能

    本文实例讲述了Android编程实现项目中异常捕获及对应Log日志文件保存功能.分享给大家供大家参考,具体如下: 做程序开发,肯定离不开与BUG打交道,更加离不开程序异常的出现.在开发的时候,我们可以通断点调试,日志打印,异常捕获工具等方式发现或处理程序中的Exception.那客户在使用我们的应用时,程序了问题,我们怎么可以知道呢?当然,我们可以加上友盟统计等第三方工具.另外还能怎么做呢?那就是把异常信息通过文档地形式保存下来,如果用户在使用的时候程序出了异常,可以让用户把对应的日志信息发给我

  • android 捕捉异常并上传至服务器的简单实现

    在项目中,我们的应用经常会遇到崩溃的情况,如果你的项目已经发送到了应用市场上,那么应用发生的崩溃开发人员是开不到的,所以我们要想办法将异常信息传到服务器上,便于开发人员查看并作出修改.Google考虑到这一点,也提供了Thread.UncaughtExceptionHandler接口来实现这一问题. 创建Crash异常捕获很简单,主要的步骤有: 1.创建BaseApplication继承Application并实现Thread.UncaughtExceptionHandler 2.通过Threa

  • Android Studio升级到3.0 Terminal 中文显示异常解决

    遇到这个问题的话应该是平时终端用的是zsh造成的,如果不是的话,那我也没办法,至少我是这么解决的,希望也可以帮到你. 具体Bug如下所示 内心有一万匹草泥马呼啸而过,捣豉了很久,最终在某群中问了下得到了答案,解决途径如下. 打开~/.zshrc文件,找到如下两行. # You may need to manually set your language environment # export LANG=en_US.UTF-8 我们把第二行取消注释即可,保存之后重启AS即可,熟悉的中文又出现在终

  • Android Studio下载更新Android SDK网络异常或无法下载

    Android Studio下载更新Android SDK网络异常或无法下载 今天重新安装了下Android Studio,在安装SDK时老是提示网络异常,于是上网查了下, 解决方法如下:HTTP Proxy中Host name设置为mirrors.neusoft.edu.cn 再勾选 这样设置就好了,试了下下载速度很快,感谢好心人提供的方法 感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

  • 安卓中出现过的一些容易被忽略的异常整理

    1.在外部开启activity时需要新开一个task,从service里开启activity时出现了这个异常. W/System.err: android.util.AndroidRuntimeException: Calling startActivity() from outside of an Activity context requires the FLAG_ACTIVITY_NEW_TASK flag. Is this really what you want? W/System.e

  • Android 中Crash时如何获取异常信息详解及实例

    Android 中Crash时如何获取异常信息详解 前言: 大家都知道,Android应用不可避免的会发生crash,无论你的程序写的多完美,总是无法完全避免crash的发生,可能是由于Android系统底层的bug,也可能是由于不充分的机型适配或者是糟糕的网络状况.当crash发生时,系统会kill掉你的程序,表现就是闪退或者程序已停止运行,这对用户来说是很不友好的,也是开发者所不愿意看到的,更糟糕的是,当用户发生了crash,开发者却无法得知程序为何crash,即便你想去解决这个crash,

随机推荐