7.0 Build.VERSION_CODES.N 24

为什么要适配

在7.0中 ,Uri.from(file),可能会触发FileUriExposedException
和动态权限一样,如果target选择24以下是不需要适配的,但还是应该尽早适配

解决办法:在24及以上,使用content:// 替代 file:// (Uri.from(file))

适配步骤

1.在res下,新建一个xml目录,在其中创建一个任意名字的xml文件,如file_provider_paths.xml

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

xml说明

external-path  等同于    Environment.getExternalStorageDirectory()
如果将path设置为path=“pic”,则共享的文件限制为/storage/emulated//pic/
为空,则共享文件目录为storage/emulated// <root-path/> 代表设备的根目录new File("/");
<files-path/> 代表context.getFilesDir()
<cache-path/> 代表context.getCacheDir()
<external-path/> 代表Environment.getExternalStorageDirectory()
<external-files-path>代表context.getExternalFilesDirs()
<external-cache-path>代表context.getExternalCacheDirs()

2.在Manifest文件中注册FileProvider对象

<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="${applicationId}.fileprovider"
android:exported="false"
android:grantUriPermissions="true"> <meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_provider_paths"/>
</provider>

3.修改java代码,场景拍照,裁剪,安装apk

 fp=context.getPackageName()+".fileprovider";

3.1 拍照

imgPath = FileUtil.generateImgePath();
File imgFile = new File(imgPath); //设置拍照后的图片的保存位置
Log.i("TAG", imgFile.getAbsolutePath());
Uri imgUri = null;
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
//通过FileProvider创建一个content类型的Uri
imgUri = FileProvider.getUriForFile(activity, fp, imgFile);
//添加这一句表示对目标应用临时授权该Uri所代表的文件
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
} else {
imgUri = Uri.fromFile(imgFile);
} intent.putExtra(MediaStore.EXTRA_OUTPUT, imgUri); final Uri uri = imgUri;
activity.startActivityForResult(intent, new CallBackIntent() {
@Override
public void onResult(Intent data) {
if (data == null || data.getData() == null)
// ToastUtils.startShort(activity, "拍照数据返回为空");
Log.i("TAG", "拍照数据返回为空"); startPhotoZoom(uri, activity, new IntentCallBack() {
@Override
public void onIntent(Intent intent) {
callBack.onBitmap(getImageView(intent));
}
});
}
});

imgPath,就是拍照后图片保存的路径

3.2 裁剪

//适配7.0的uri,(文件uri转化为ContentUri)
public Uri getImageContentUri(Context context, File imageFile) {
String filePath = imageFile.getAbsolutePath();
Cursor cursor = context.getContentResolver().query(
MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
new String[] { MediaStore.Images.Media._ID },
MediaStore.Images.Media.DATA + "=? ",
new String[] { filePath }, null); if (cursor != null && cursor.moveToFirst()) {
int id = cursor.getInt(cursor
.getColumnIndex(MediaStore.MediaColumns._ID));
Uri baseUri = Uri.parse("content://media/external/images/media");
return Uri.withAppendedPath(baseUri, "" + id);
} else {
if (imageFile.exists()) {
ContentValues values = new ContentValues();
values.put(MediaStore.Images.Media.DATA, filePath);
return context.getContentResolver().insert(
MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);
} else {
return null;
}
}
}

如果是由相机或者相册之后调用裁减,无需转化uri
如果是单独使用裁减功能,则需要利用上述方法进行转化

private void cropPicture(Uri uri) {
Intent innerIntent = new Intent("com.android.camera.action.CROP");
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
//添加这一句表示对目标应用临时授权该Uri所代表的文件
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
}
innerIntent.setDataAndType(uri, "image/*");
innerIntent.putExtra("crop", "true"); //不是1比1 主要是为了适配华为手机一比一时,裁减是圆形的
innerIntent.putExtra("aspectX", );
innerIntent.putExtra("aspectY", ); innerIntent.putExtra("outputX", );
innerIntent.putExtra("outputY", );
innerIntent.putExtra("return-data", true);
innerIntent.putExtra("scale", true); activity.startActivityForResult(intent, new CallBackIntent() {
@Override
public void onResult(Intent data) {
callBack.onIntent(data);
//裁减后的图片
//Bitmap bitmap = data.getParcelableExtra("data");
}
}); }

3.3 安装apk

private void installAPK(UpdateInfo updateInfo) {
File apkfile = new File(mSavePath, updateInfo.getClientVersionName());
if (!apkfile.exists()) {
return;
}
String cmd = "chmod 777 " + apkfile.toString();
try {
Runtime.getRuntime().exec(cmd);
} catch (Exception e) {
e.printStackTrace();
}
// 通过Intent安装APK文件
Intent i = new Intent(Intent.ACTION_VIEW);
if (Build.VERSION.SDK_INT >= ) { //适配安卓7.0
i.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
Uri apkFileUri = FileProvider.getUriForFile(context.getApplicationContext(), fp, apkfile);
i.setDataAndType(apkFileUri, "application/vnd.android.package-archive");
} else {
i.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
i.setDataAndType(Uri.parse("file://" + apkfile.toString()),"application/vnd.android.package-archive");
}
context.startActivity(i);
//关闭当前程序
int pid = android.os.Process.myPid();
android.os.Process.killProcess(pid);
}

7.0 FileProvide适配的更多相关文章

  1. Android 6.0权限适配

    targetSdkVersion 23以上,必须适配新的权限模式 安卓6.0及之后,权限分为三类  1.不涉及隐私的正常权限,如innernet2.危险权限 3.特殊权限 system_alert_w ...

  2. Android项目实战(四十九):Andoird 7.0+相机适配

    解决方案类似: Android项目实战(四十):Andoird 7.0+ 安装APK适配 解决方法: 一.在AndroidManifest.xml 文件中添加 四大组件之一的 <provider ...

  3. 【Android】我有放入Icon到mipmap,但不显示,只显示安卓机器人Icon(Android 8.0 图标适配)

    首先,放上别人写的博客,而我自己的博客,只会写大概思路,给自己留给备忘 https://blog.csdn.net/guolin_blog/article/details/79417483 其实会发生 ...

  4. GreenDao2.2升级GreenDao3.0的适配之路

    前言.为什么要升级到Greendao3.0? 1. 多人开发 以往的数据库建表建Dao等操作要新开一个module,在统一的地方管理数据库建表,现在可以直接写Entity.多人开发时自己管自己的Ent ...

  5. android9.0系统适配遇到的问题

    一.apk在9.0以下的系统上安装运行,没有问题.但是在9.0系统上运行会弹出一个框 解决办法: private void closeAndroidPDialog() { try { Class aC ...

  6. [转]cocos2d-js 3.0 屏幕适配方案 分辨率适应

    首先介绍一个api和相应的参数: cc.view.setDesignResolutionSize(1024, 768, cc.ResolutionPolicy.FIXED_WIDTH); 这里设置游戏 ...

  7. cocos2d-js 3.0 屏幕适配方案 分辨率适应

    首先介绍一个api和相应的参数: cc.view.setDesignResolutionSize(1024, 768, cc.ResolutionPolicy.FIXED_WIDTH); 这里设置游戏 ...

  8. Win10《芒果TV》商店版更新v3.1.4.0:适配Xbox手柄B键后退、手机支持暗色主题不伤眼

    在双十一全球剁手节.光棍节欢庆之际,<芒果TV>UWP版迅速更新v3.1.4版,适配Xbox手柄B键全局后退,支持手机切换暗色主题,优化并解决启动卡顿等问题. 芒果TV UWP V3.1. ...

  9. Android权限管理之RxPermission解决Android 6.0 适配问题

    前言: 上篇重点学习了Android 6.0的运行时权限,今天还是围绕着Android 6.0权限适配来总结学习,这里主要介绍一下我们公司解决Android 6.0权限适配的方案:RxJava+RxP ...

随机推荐

  1. opencv实现正交匹配追踪算法OMP

    //dic: 字典矩阵: //signal :待重构信号(一次只能重构一个信号,即一个向量) //min_residual: 最小残差 //sparsity:稀疏度 //coe:重构系数 //atom ...

  2. Redis学习系列三List列表

    一.简介 Redis中的列表相当于C#中的LinkedList,也就是链表,如果你研究过链表这个数据结构,肯定知道.它的插入和删除是非常快的,但是定位却很慢,因为必须遍历所有的元素,才能找到对应的值, ...

  3. Android get current Locale, not default

    he default Locale is constructed statically at runtime for your application process from the system ...

  4. 第k大的数

    问题描述:输入一组数,指定一个k,输出这组数里第k大的数. 一般这种题目,第一想法是把整个数组先排序后,再选取第k位的数.但是这样做实际上浪费了大量的时间在排序上,我们只是要求第k大的数,并非要把整个 ...

  5. Hadoop2源码分析-RPC机制初识

    1.概述 上一篇博客,讲述Hadoop V2的序列化机制,这为我们学习Hadoop V2的RPC机制奠定了基础.RPC的内容涵盖的信息有点多,包含Hadoop的序列化机制,RPC,代理,NIO等.若对 ...

  6. CentOS7 下安装 iSCSI Target(tgt) ,使用 Ceph rbd

    目录 一.iSCSI 介绍 1. iSCSI 定义 2. 几种常见的 iSCSI Target 3. 优缺点比较 二.安装步骤 1. 关闭防火墙 2. 关闭selinux 3. 通过 yum 安装 t ...

  7. PTA (Advanced Level) 1005 Spell It Right

    Spell It Right Given a non-negative integer N, your task is to compute the sum of all the digits of  ...

  8. mysql LAST_INSERT_ID详解

    http://blog.sina.com.cn/s/blog_5b5460eb0100nwvo.html LAST_INSERT_ID() LAST_INSERT_ID(expr) 自动返回最后一个I ...

  9. 各大语言性能对比PK数据

    这里我用的python版本是Python 3.6.2(64位),php版本是PHP 7.0.12(64位),node版本是v6.11.0(64位),Go 1.8.3(64位),C# 基于.Net 4. ...

  10. 解决MVC应用程序数据重复加载问题

    先来看看这个动画: 这是使用jQuery来实现数据加载,每点击一次,数据就加载一次.这源程序与实现来自<MVC应用程序JsonResult()的练习>http://www.cnblogs. ...