工具类FileUtils :

import android.annotation.SuppressLint;
import android.content.ContentUris;
import android.content.Context;
import android.database.Cursor;
import android.net.Uri;
import android.os.Environment;
import android.os.Build.VERSION;
import android.provider.DocumentsContract;
import android.provider.MediaStore.Images.Media;
import android.text.TextUtils;
import android.util.Log;
import androidx.annotation.NonNull;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.nio.channels.FileChannel;
import java.text.SimpleDateFormat;
import java.util.Locale;public class FileUtils {private static final String TAG = "FileUtils";private static SimpleDateFormat sf = new SimpleDateFormat("yyyyMMdd_HHmmssSS");private FileUtils() {}public static boolean isExternalStorageDocument(Uri uri) {return "com.android.externalstorage.documents".equals(uri.getAuthority());}public static boolean isDownloadsDocument(Uri uri) {return "com.android.providers.downloads.documents".equals(uri.getAuthority());}public static boolean isMediaDocument(Uri uri) {return "com.android.providers.media.documents".equals(uri.getAuthority());}public static boolean isGooglePhotosUri(Uri uri) {return "com.google.android.apps.photos.content".equals(uri.getAuthority());}public static String getDataColumn(Context context, Uri uri, String selection, String[] selectionArgs) {Cursor cursor = null;String column = "_data";String[] projection = new String[]{"_data"};try {cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs, (String)null);if (cursor != null && cursor.moveToFirst()) {int column_index = cursor.getColumnIndexOrThrow("_data");String var8 = cursor.getString(column_index);return var8;}} catch (IllegalArgumentException var12) {Log.i("FileUtils", String.format(Locale.getDefault(), "getDataColumn: _data - [%s]", var12.getMessage()));} finally {if (cursor != null) {cursor.close();}}return null;}@SuppressLint({"NewApi"})public static String getPath(Context context, Uri uri) {boolean isKitKat = VERSION.SDK_INT >= 19;if (isKitKat && DocumentsContract.isDocumentUri(context, uri)) {String id;String[] split;String type;if (isExternalStorageDocument(uri)) {id = DocumentsContract.getDocumentId(uri);split = id.split(":");type = split[0];if ("primary".equalsIgnoreCase(type)) {return Environment.getExternalStorageDirectory() + "/" + split[1];}} else if (isDownloadsDocument(uri)) {id = DocumentsContract.getDocumentId(uri);if (!TextUtils.isEmpty(id)) {try {Uri contentUri = ContentUris.withAppendedId(Uri.parse("content://downloads/public_downloads"), Long.valueOf(id));return getDataColumn(context, contentUri, (String)null, (String[])null);} catch (NumberFormatException var9) {Log.i("FileUtils", var9.getMessage());return null;}}} else if (isMediaDocument(uri)) {id = DocumentsContract.getDocumentId(uri);split = id.split(":");type = split[0];Uri contentUri = null;if ("image".equals(type)) {contentUri = Media.EXTERNAL_CONTENT_URI;} else if ("video".equals(type)) {contentUri = android.provider.MediaStore.Video.Media.EXTERNAL_CONTENT_URI;} else if ("audio".equals(type)) {contentUri = android.provider.MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;}String selection = "_id=?";String[] selectionArgs = new String[]{split[1]};return getDataColumn(context, contentUri, "_id=?", selectionArgs);}} else {if ("content".equalsIgnoreCase(uri.getScheme())) {if (isGooglePhotosUri(uri)) {return uri.getLastPathSegment();}return getDataColumn(context, uri, (String)null, (String[])null);}if ("file".equalsIgnoreCase(uri.getScheme())) {return uri.getPath();}}return null;}public static boolean copyFile(FileInputStream fileInputStream, String outFilePath) throws IOException {if (fileInputStream == null) {return false;} else {FileChannel inputChannel = null;FileChannel outputChannel = null;boolean var5;try {inputChannel = fileInputStream.getChannel();outputChannel = (new FileOutputStream(new File(outFilePath))).getChannel();inputChannel.transferTo(0L, inputChannel.size(), outputChannel);inputChannel.close();boolean var4 = true;return var4;} catch (Exception var9) {var5 = false;} finally {if (fileInputStream != null) {fileInputStream.close();}if (inputChannel != null) {inputChannel.close();}if (outputChannel != null) {outputChannel.close();}}return var5;}}public static void copyFile(@NonNull String pathFrom, @NonNull String pathTo) throws IOException {if (!pathFrom.equalsIgnoreCase(pathTo)) {FileChannel outputChannel = null;FileChannel inputChannel = null;try {inputChannel = (new FileInputStream(new File(pathFrom))).getChannel();outputChannel = (new FileOutputStream(new File(pathTo))).getChannel();inputChannel.transferTo(0L, inputChannel.size(), outputChannel);inputChannel.close();} finally {if (inputChannel != null) {inputChannel.close();}if (outputChannel != null) {outputChannel.close();}}}}public static String getCreateFileName(String prefix) {long millis = System.currentTimeMillis();return prefix + sf.format(millis);}public static String getCreateFileName() {long millis = System.currentTimeMillis();return sf.format(millis);}public static String rename(String fileName) {String temp = fileName.substring(0, fileName.lastIndexOf("."));String suffix = fileName.substring(fileName.lastIndexOf("."));return temp + "_" + getCreateFileName() + suffix;}
}

一、第一种实现

/*** 根据Uri获取文件绝对路径,解决Android4.4以上版本Uri转换 兼容Android 10** @param context* @param imageUri*/public static String getFileAbsolutePath(Context context, Uri imageUri) {if (context == null || imageUri == null) {return null;}if (android.os.Build.VERSION.SDK_INT < android.os.Build.VERSION_CODES.KITKAT) {return getRealFilePath(context, imageUri);}if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.KITKAT && android.os.Build.VERSION.SDK_INT < Build.VERSION_CODES.Q && DocumentsContract.isDocumentUri(context, imageUri)) {if (isExternalStorageDocument(imageUri)) {String docId = DocumentsContract.getDocumentId(imageUri);String[] split = docId.split(":");String type = split[0];if ("primary".equalsIgnoreCase(type)) {return Environment.getExternalStorageDirectory() + "/" + split[1];}} else if (isDownloadsDocument(imageUri)) {String id = DocumentsContract.getDocumentId(imageUri);Uri contentUri = ContentUris.withAppendedId(Uri.parse("content://downloads/public_downloads"), Long.valueOf(id));return getDataColumn(context, contentUri, null, null);} else if (isMediaDocument(imageUri)) {String docId = DocumentsContract.getDocumentId(imageUri);String[] split = docId.split(":");String type = split[0];Uri contentUri = null;if ("image".equals(type)) {contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;} else if ("video".equals(type)) {contentUri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI;} else if ("audio".equals(type)) {contentUri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;}String selection = MediaStore.Images.Media._ID + "=?";String[] selectionArgs = new String[]{split[1]};return getDataColumn(context, contentUri, selection, selectionArgs);}}// MediaStore (and general)if (android.os.Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q){return uriToFileApiQ(context,imageUri);}else if ("content".equalsIgnoreCase(imageUri.getScheme())) {// Return the remote addressif (isGooglePhotosUri(imageUri)) {return imageUri.getLastPathSegment();}return getDataColumn(context, imageUri, null, null);}// Fileelse if ("file".equalsIgnoreCase(imageUri.getScheme())) {return imageUri.getPath();}return null;}//此方法 只能用于4.4以下的版本private static String getRealFilePath(final Context context, final Uri uri) {if (null == uri) {return null;}final String scheme = uri.getScheme();String data = null;if (scheme == null) {data = uri.getPath();} else if (ContentResolver.SCHEME_FILE.equals(scheme)) {data = uri.getPath();} else if (ContentResolver.SCHEME_CONTENT.equals(scheme)) {String[] projection = {MediaStore.Images.ImageColumns.DATA};Cursor cursor = context.getContentResolver().query(uri, projection, null, null, null);//            Cursor cursor = context.getContentResolver().query(uri, new String[]{MediaStore.Images.ImageColumns.DATA}, null, null, null);if (null != cursor) {if (cursor.moveToFirst()) {int index = cursor.getColumnIndex(MediaStore.Images.ImageColumns.DATA);if (index > -1) {data = cursor.getString(index);}}cursor.close();}}return data;}/*** @param uri The Uri to check.* @return Whether the Uri authority is ExternalStorageProvider.*/private static boolean isExternalStorageDocument(Uri uri) {return "com.android.externalstorage.documents".equals(uri.getAuthority());}/*** @param uri The Uri to check.* @return Whether the Uri authority is DownloadsProvider.*/private static boolean isDownloadsDocument(Uri uri) {return "com.android.providers.downloads.documents".equals(uri.getAuthority());}private static String getDataColumn(Context context, Uri uri, String selection, String[] selectionArgs) {Cursor cursor = null;String column = MediaStore.Images.Media.DATA;String[] projection = {column};try {cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs, null);if (cursor != null && cursor.moveToFirst()) {int index = cursor.getColumnIndexOrThrow(column);return cursor.getString(index);}} finally {if (cursor != null) {cursor.close();}}return null;}/*** @param uri The Uri to check.* @return Whether the Uri authority is MediaProvider.*/private static boolean isMediaDocument(Uri uri) {return "com.android.providers.media.documents".equals(uri.getAuthority());}/*** @param uri The Uri to check.* @return Whether the Uri authority is Google Photos.*/private static boolean isGooglePhotosUri(Uri uri) {return "com.google.android.apps.photos.content".equals(uri.getAuthority());}/*** Android 10 以上适配 另一种写法* @param context* @param uri* @return*/private static String getFileFromContentUri(Context context, Uri uri) {if (uri == null) {return null;}String filePath;String[] filePathColumn = {MediaStore.MediaColumns.DATA, MediaStore.MediaColumns.DISPLAY_NAME};ContentResolver contentResolver = context.getContentResolver();Cursor cursor = contentResolver.query(uri, filePathColumn, null,null, null);if (cursor != null) {cursor.moveToFirst();try {filePath = cursor.getString(cursor.getColumnIndex(filePathColumn[0]));return filePath;} catch (Exception e) {} finally {cursor.close();}}return "";}/*** Android 10 以上适配* @param context* @param uri* @return*/@RequiresApi(api = Build.VERSION_CODES.Q)private static String uriToFileApiQ(Context context, Uri uri) {File file = null;//android10以上转换if (uri.getScheme().equals(ContentResolver.SCHEME_FILE)) {file = new File(uri.getPath());} else if (uri.getScheme().equals(ContentResolver.SCHEME_CONTENT)) {//把文件复制到沙盒目录ContentResolver contentResolver = context.getContentResolver();Cursor cursor = contentResolver.query(uri, null, null, null, null);if (cursor.moveToFirst()) {String displayName = cursor.getString(cursor.getColumnIndex(OpenableColumns.DISPLAY_NAME));try {InputStream is = contentResolver.openInputStream(uri);File cache = new File(context.getExternalCacheDir().getAbsolutePath(), Math.round((Math.random() + 1) * 1000) + displayName);FileOutputStream fos = new FileOutputStream(cache);FileUtils.copy(is, fos);file = cache;fos.close();is.close();} catch (IOException e) {e.printStackTrace();}}}return file.getAbsolutePath();}/*** 通过文件路径 uri的转字符也可以* @param filePath* @return*/public static String getMimeType(String filePath) {String ext = MimeTypeMap.getFileExtensionFromUrl(filePath);return MimeTypeMap.getSingleton().getMimeTypeFromExtension(ext);}

二、第二种通用实现(目前文件选择贼好用-部分需大家测试验证,安卓10及以下使用无问题,11,12官方有对应的区分)

/*** Android 10 以上适配 另一种写法* @param context* @param uri* @return*/@SuppressLint("Range")public static String getFileFromContentUri(Context context, Uri uri) {if (uri == null) {return null;}String filePath;String[] filePathColumn = {MediaStore.DownloadColumns.DATA, MediaStore.DownloadColumns.DISPLAY_NAME};ContentResolver contentResolver = context.getContentResolver();Cursor cursor = contentResolver.query(uri, filePathColumn, null,null, null);if (cursor != null) {cursor.moveToFirst();try {filePath = cursor.getString(cursor.getColumnIndex(filePathColumn[0]));return filePath;} catch (Exception e) {} finally {cursor.close();}}return "";}

新方式兼容安卓13通用实现:

public static String getRealPathFromUri(Context context, Uri uri) {String filePath = "";String scheme = uri.getScheme();if (scheme == null)filePath = uri.getPath();else if (ContentResolver.SCHEME_FILE.equals(scheme)) {filePath = uri.getPath();} else if (ContentResolver.SCHEME_CONTENT.equals(scheme)) {String[] proj = { MediaStore.Images.Media.DATA };Cursor cursor = context.getContentResolver().query(uri, proj, null, null, null);if (cursor != null) {if (cursor.moveToFirst()) {int columnIndex = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);filePath = cursor.getString(columnIndex);}cursor.close();}if (TextUtils.isEmpty(filePath)) {filePath = getFilePathForNonMediaUri(context, uri);}}return filePath;}//非媒体文件中查找private static String getFilePathForNonMediaUri(Context context, Uri uri) {String filePath = "";Cursor cursor = context.getContentResolver().query(uri, null, null, null, null);if (cursor != null) {if (cursor.moveToFirst()) {int columnIndex = cursor.getColumnIndexOrThrow("_data");filePath = cursor.getString(columnIndex);}cursor.close();}return filePath;}

三 、Path 转Uri

    public static Uri toUri(Context context,String filePath) {if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {return FileProvider.getUriForFile(context, context.getApplicationInfo().packageName + ".fileprovider", new File(filePath));}return Uri.fromFile(new File(filePath));}

四,file_paths文件

<?xml version="1.0" encoding="utf-8"?>
<resources><paths><root-pathname="root"path="" /><files-pathname="files"path="" /><cache-pathname="cache"path="" /><external-pathname="external"path="" /><external-files-pathname="external_file_path"path="" /><external-cache-pathname="external_cache_path"path="" /></paths>
</resources>

五 、mainfest 文件配置

<providerandroid:name="androidx.core.content.FileProvider"android:authorities="你的包名.fileprovider"android:exported="false"android:grantUriPermissions="true"tools:replace="android:authorities"><meta-dataandroid:name="android.support.FILE_PROVIDER_PATHS"android:resource="@xml/file_paths" /></provider>

六、Android 10 的坑:

从Android10(API 29)开始,App层没有访问此路径的权限(即:

Environment.getExternalStorageDirectory().getAbsolutePath()

),官方提供的临时解决方案是在 Android Manifest 清单文件添加如下配置 

七、Android 11 的坑:

需要增加:Manifest.permission.MANAGE_EXTERNAL_STORAGE 权限

步骤:

  1. 在AndroidManifest.xml中声明MANAGE_EXTERNAL_STORAGE权限。

  2. 发出一个action为Settings.ACTION_MANAGE_ALL_FILES_ACCESS_PERMISSION的Intent,引导用户手动授权。

  3. 调用Environment.isExternalStorageManager()来判断用户是否已授权。

Android10/11/12/13 Uri与真实路径转换--全适配相关推荐

  1. android11以上 uri转file uri转真实路径

    很久以前,曾经写过针对anddroid10 uri转file的方法:android10以上 uri转file uri转真实路径_jingzz1的博客-CSDN博客_android uri转filefu ...

  2. PTA浙大版python程序设计题目集--第2章-1 计算 11+12+13+...+m (30 分)

    输入一个正整数m(20<=m<=100),计算 11+12+13+-+m 的值. 输入格式: 在一行输入一个正整数m. 输出格式: 在一行中按照格式"sum = S"输 ...

  3. python 整数输出 d f_如何将数字(10,11,12,13,14,15)分配给Python 3中的字母(A,B,C,D,E,F)?...

    您可以在代码中添加更多行来执行此操作: 首先创建两个带有字符的列表,一个带有要映射的整数,然后从那些创建dict: list_1=["A","B"," ...

  4. 苹果手机用什么软件测试续航,苹果哪个系统最耗电 iOS10/11/12/13电池续航测试

    自从iOS13发布以后,苹果多次更新版本,但故障还是很多.而且在之前的测试当中,iOS13.2的性能与电池续航表现都非常糟糕,还不如不升级呢?那么过去几年当中,哪版iOS系统电池续航表现最好呢?接下来 ...

  5. 第2章-1 计算 11+12+13+...+m

    第2章-1 计算 11+12+13+-+m 输入一个正整数m(20<=m<=100),计算 11+12+13+-+m 的值. 输入格式: 在一行输入一个正整数m. 输出格式: 在一行中按照 ...

  6. 计算 11+12+13+...+m python

    第2章-1 计算 11+12+13+-+m (30 分) 输入一个正整数m(20<=m<=100),计算 11+12+13+-+m 的值. 输入格式: 在一行输入一个正整数m. 输出格式: ...

  7. 鸟哥的Linux私房菜-第10/11/12/13章(vim程序编辑器、学习bash、正则表达式与文件格式化处理、学习Shell Scripts)...

    第10章 vim程序编辑器 可以将vim看做vi的进阶版本,vim可以用颜色或底线等方式来显示出一些特殊的信息. 为何要学习vim?因为: a. 所有的 Unix Like 系统都会内建 vi 文书编 ...

  8. 【传智播客】Javaweb程序设计任务教程 黑马程序员 第7,8,9,10,11,12,13,14,15章 课后答案

    所有章节答案合集-->传送门 第一部分是题目,向下翻, 第二部分是答案 第七章 [测一测] 学习完前面的内容,下面来动手测一测吧,请思考以下问题: 1.简述Javabean规范都有哪些?(至少写 ...

  9. jdk 8 、9 10 11 12 13 14和 jdk 1.8 什么关系??

    五.jdk 13 和 jdk 1.8 什么关系 jdk 8 就是 jdk 1.8 jdk9 就是 jdk 1.9 其他同理 官方虽然更新的快,但是大多数公司,为因为习惯问题和调整的麻烦,在加上 jdk ...

最新文章

  1. [Python学习] 专题二.条件语句和循环语句的基础知识
  2. 信息学奥赛一本通 1985:【19CSPJ普及组】加工零件 | 洛谷 P5663 [CSP-J2019] 加工零件
  3. 逆向课程第二讲,寻找main入口点
  4. HOWTO:InstallShield中如何通过脚本获取“My Documents”路径
  5. Android开发系列(十一) QQ登陆界面——Android控件使用实例
  6. android获取webview url,android - webview获取到当前页面的url
  7. AgentWeb url监听问题
  8. 内存的管理方式有哪些
  9. 为什么还有6亿中国人仍不放弃QQ?
  10. linux 查看vcf文件,VCF格式文件的shell小练习
  11. 【学数据结构】-----串(顺序串、堆串、块链串)(7000字总结+代码+图)
  12. 实时互联网的隐形风口
  13. OpenCV_basis
  14. 联想微型计算机供电电源线,拆修一只联想电源适配器,告诉你一个不为人知的秘密...
  15. 职场神攻略:5分钟自我介绍法 快准狠!
  16. odoo中的消息通知
  17. 拟人肌肉骨骼气动人工肌肉驱动的10自由度机器人手臂
  18. Windows10 应用商店打不开问题,报错:Code: 0x80072F7D
  19. 谨以此篇文章开启我的博客生涯
  20. NOI 模拟试题(一)

热门文章

  1. 二分图最大匹配(匈牙利算法Dinic算法)
  2. 为什么要使用接口和抽象类?
  3. unity3d 滚动字幕
  4. 支付宝直连商户开通步骤
  5. 考导游证需要什么条件?报考导游领队需要什么条件?
  6. 背景色及背景图片的覆盖范围
  7. linux系统LVM磁盘扩容
  8. vue项目打包后部署到服务器(超详细步骤)
  9. 电子劳动合同如何签订才能生效呢?四步就能教会你!
  10. java编译器——安装IDEA 步骤