调用系统图库会出现两种uri的问题,一个是在文件管理器中的图库中,获取到的地址为:content://media/external/images/media/972  这种格式

另外一种的是系统文件管理器的图片选选项中,获取到的路径格式为:content://com.android.providers.media.documents/document/image%3A972

听别人解释,这两种路径,第一种是下面这种的映射,也就是根据第一种可以拿到第二种,但个人还是不太明白。看到的高人请指点啊。

这两种uri的主要问题出在,如果设置图片使用  setImageUri(uri) 时,二者都没问题。

但是如果要把该uri转换成图片的路径,下面的长的路径转化出来为null。上面的ok。

我的转化方法如下:

Uri uri = data.getData();
    
ContentResolver resolver = getContentResolver();

Cursor cursor = resolver.query(uri,
   
new String[]{MediaStore.Images.Media.DATA}, null, null, null);
cursor.moveToFirst();
String path = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA));

以下为引用别人博客

出处:http://blog.csdn.net/tempersitu/article/details/20557383

  1. Intent intent=new Intent(Intent.ACTION_PICK, android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI);

的 方式来做,是调用系统图库来做,但是发现如果有图片是同步到google相册的话,图库里面能看到一个auto backup的目录,点进去选图片的话是无法获取到图片的路径的.因为那些图片根本就不存在于手机上.然后看到无论是百度贴吧,Instagram,或者 还有些会选取图片做修改的app,都是用一个很漂亮的图片选择器(4.4以上,4.3的还是用系统旧的图库).

而这个图片选择器可以屏蔽掉那个auto
backup的目录.所以就开始打算用这个图片选择器来选图片了.
这个方法就是

  1. Intent intent=new Intent(Intent.ACTION_GET_CONTENT);//ACTION_OPEN_DOCUMENT
  2. intent.addCategory(Intent.CATEGORY_OPENABLE);
  3. intent.setType("image/jpeg");
  4. if(android.os.Build.VERSION.SDK_INT>=android.os.Build.VERSION_CODES.KITKAT){
  5. startActivityForResult(intent, SELECT_PIC_KITKAT);
  6. }else{
  7. startActivityForResult(intent, SELECT_PIC);
  8. }

为什么要分开不同版本呢?其实在4.3或以下可以直接用ACTION_GET_CONTENT的,在4.4或以上,官方建议用ACTION_OPEN_DOCUMENT,但其实都不算太大区别,区别是他们返回的Uri,那个才叫大区别.这就是困扰了我一整天的问题所在了.

4.3或以下,选了图片之后,根据Uri来做处理,很多帖子都有了,我就不详细说了.主要是4.4,如果使用上面pick的原生方法来选图,返回的uri
还是正常的,但如果用ACTION_GET_CONTENT的方法,返回的uri跟4.3是完全不一样的,4.3返回的是带文件路径的,而4.4返回的却
是content://com.android.providers.media.documents/document/image:3951这样的,
没有路径,只有图片编号的uri.这就导致接下来无法根据图片路径来裁剪的步骤了.

还好找了很多方法,包括加权限啊什么的,中间还试过用一些方法,自己的app没崩溃,倒是让系统图库崩溃了,引发了java.lang.SecurityException.

  1. Caused by: java.lang.SecurityException: Permission Denial: opening provider com.android.providers.media.MediaDocumentsProvider from ProcessRecord{437b5d88 9494:com.google.android.gallery3d/u0a20} (pid=9494, uid=10020) requires android.permission.MANAGE_DOCUMENTS or android.permission.MANAGE_DOCUMENTS

看来4.4的系统还是有些bug.重点来了,4.4得到的uri,需要以下方法来获取文件的路径

  1. public static String getPath(final Context context, final Uri uri) {
  2. final boolean isKitKat = Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT;
  3. // DocumentProvider
  4. if (isKitKat && DocumentsContract.isDocumentUri(context, uri)) {
  5. // ExternalStorageProvider
  6. if (isExternalStorageDocument(uri)) {
  7. final String docId = DocumentsContract.getDocumentId(uri);
  8. final String[] split = docId.split(":");
  9. final String type = split[0];
  10. if ("primary".equalsIgnoreCase(type)) {
  11. return Environment.getExternalStorageDirectory() + "/" + split[1];
  12. }
  13. // TODO handle non-primary volumes
  14. }
  15. // DownloadsProvider
  16. else if (isDownloadsDocument(uri)) {
  17. final String id = DocumentsContract.getDocumentId(uri);
  18. final Uri contentUri = ContentUris.withAppendedId(
  19. Uri.parse("content://downloads/public_downloads"), Long.valueOf(id));
  20. return getDataColumn(context, contentUri, null, null);
  21. }
  22. // MediaProvider
  23. else if (isMediaDocument(uri)) {
  24. final String docId = DocumentsContract.getDocumentId(uri);
  25. final String[] split = docId.split(":");
  26. final String type = split[0];
  27. Uri contentUri = null;
  28. if ("image".equals(type)) {
  29. contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
  30. } else if ("video".equals(type)) {
  31. contentUri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI;
  32. } else if ("audio".equals(type)) {
  33. contentUri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
  34. }
  35. final String selection = "_id=?";
  36. final String[] selectionArgs = new String[] {
  37. split[1]
  38. };
  39. return getDataColumn(context, contentUri, selection, selectionArgs);
  40. }
  41. }
  42. // MediaStore (and general)
  43. else if ("content".equalsIgnoreCase(uri.getScheme())) {
  44. // Return the remote address
  45. if (isGooglePhotosUri(uri))
  46. return uri.getLastPathSegment();
  47. return getDataColumn(context, uri, null, null);
  48. }
  49. // File
  50. else if ("file".equalsIgnoreCase(uri.getScheme())) {
  51. return uri.getPath();
  52. }
  53. return null;
  54. }
  55. /**
  56. * Get the value of the data column for this Uri. This is useful for
  57. * MediaStore Uris, and other file-based ContentProviders.
  58. *
  59. * @param context The context.
  60. * @param uri The Uri to query.
  61. * @param selection (Optional) Filter used in the query.
  62. * @param selectionArgs (Optional) Selection arguments used in the query.
  63. * @return The value of the _data column, which is typically a file path.
  64. */
  65. public static String getDataColumn(Context context, Uri uri, String selection,
  66. String[] selectionArgs) {
  67. Cursor cursor = null;
  68. final String column = "_data";
  69. final String[] projection = {
  70. column
  71. };
  72. try {
  73. cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs,
  74. null);
  75. if (cursor != null && cursor.moveToFirst()) {
  76. final int index = cursor.getColumnIndexOrThrow(column);
  77. return cursor.getString(index);
  78. }
  79. } finally {
  80. if (cursor != null)
  81. cursor.close();
  82. }
  83. return null;
  84. }
  85. /**
  86. * @param uri The Uri to check.
  87. * @return Whether the Uri authority is ExternalStorageProvider.
  88. */
  89. public static boolean isExternalStorageDocument(Uri uri) {
  90. return "com.android.externalstorage.documents".equals(uri.getAuthority());
  91. }
  92. /**
  93. * @param uri The Uri to check.
  94. * @return Whether the Uri authority is DownloadsProvider.
  95. */
  96. public static boolean isDownloadsDocument(Uri uri) {
  97. return "com.android.providers.downloads.documents".equals(uri.getAuthority());
  98. }
  99. /**
  100. * @param uri The Uri to check.
  101. * @return Whether the Uri authority is MediaProvider.
  102. */
  103. public static boolean isMediaDocument(Uri uri) {
  104. return "com.android.providers.media.documents".equals(uri.getAuthority());
  105. }
  106. /**
  107. * @param uri The Uri to check.
  108. * @return Whether the Uri authority is Google Photos.
  109. */
  110. public static boolean isGooglePhotosUri(Uri uri) {
  111. return "com.google.android.apps.photos.content".equals(uri.getAuthority());
  112. }

这样,就可以在4.4上用漂亮的图片选择器,选到我们想要的文件,又不会出问题了.

昨天发现了个bug,如果在4.4上面不用"图片"来选,用"图库"来选,就会无法读取到图片路径,所以只需要加个判断,如果是用旧方式来选,就用旧方式来读,就是如果
DocumentsContract.isDocumentUri(context, uri) 返回false的话,就用旧的方式

    1. public static String selectImage(Context context,Intent data){
    2. Uri selectedImage = data.getData();
    3. //      Log.e(TAG, selectedImage.toString());
    4. if(selectedImage!=null){
    5. String uriStr=selectedImage.toString();
    6. String path=uriStr.substring(10,uriStr.length());
    7. if(path.startsWith("com.sec.android.gallery3d")){
    8. Log.e(TAG, "It's auto backup pic path:"+selectedImage.toString());
    9. return null;
    10. }
    11. }
    12. String[] filePathColumn = { MediaStore.Images.Media.DATA };
    13. Cursor cursor = context.getContentResolver().query(selectedImage,filePathColumn, null, null, null);
    14. cursor.moveToFirst();
    15. int columnIndex = cursor.getColumnIndex(filePathColumn[0]);
    16. String picturePath = cursor.getString(columnIndex);
    17. cursor.close();
    18. return picturePath;
    19. }

系统图片uri的问题的更多相关文章

  1. Android系统常用URI

    android系统常用URI android系统管理联系人的URI如下: ContactsContract.Contacts.CONTENT_URI 管理联系人的Uri ContactsContrac ...

  2. 适配Android4.4~Android11,调用系统相机,系统相册,系统图片裁剪,转换文件(对图片进行上传等操作)

    前言 最近Android对于文件的许多方法进行了修改,网络上又没有对Android4到Android11关于系统相机.系统相册和系统裁剪的适配方案,我花了几天事件总结了一下,先上源码 DEMO源码 先 ...

  3. 根据图片Uri获得图片文件

    2013-12-17 1. 根据联系人图片Uri获得图片文件并将它显示在ImageView上, 代码如下: Uri uri = Uri.parse("content://com.androi ...

  4. WPF应用加载图片URI指定需要注意的地方

    应用程序(.exe)加载图片: 可以省略"pack://application:,,," 打头,因为系统运行时需要的图片文件在Exe程序集(组合体)中:譬如: <Image ...

  5. android调用系统图片浏览器裁切后出现黑边

    是这样的:我使用系统的图片浏览器,然后让它自动跳到图片裁切界面,当我们定义了返回的图片大小过大,而我们实际的图片像素达不到时,系统为我们自动地填充了不够的像素成黑色,那么我们怎么样来解决这个问题呢?不 ...

  6. Android根据图片Uri获取图片path绝对路径的几种方法【转】

    在Android 编程中经常会用到Uri转化为文件路径,如我们从相册选择图片上传至服务器,一般上传前需要对图片进行压缩,这时候就要用到图片的绝对路径. 下面对我开发中uri转path路径遇到的问题进行 ...

  7. iOS 取得单张系统图片

    这里主要用到了UIImagePickerController 不多废话,直接上代码 // // RootViewController.m // GetImageFromPhotoAlbum // // ...

  8. IOS 横屏中添加UIImagePickerController获取系统图片

    今天写ipad的项目,然后需要调用系统相册选择图片,然后用了UIImagePickerController ,崩溃了,后来查了一下,UIImagePickerController只支持竖屏,但是... ...

  9. Android RatingBar自定义替换系统图片

    1.自定义替换提醒☆图片,准备两个图片添加到系统中去:如下:        在drewable下定义一个图片资源ratingbar_drawable.xml 1 2 3 4 5 6 7 8 9 10 ...

随机推荐

  1. web文件上传的实现

    1,html页面,上传使用input type=file控件,其所在的form必须加上enctype="multipart/form-data" <form role=&qu ...

  2. jQuery中append()与appendto()用法分析

    本文实例分析了jquery中append()与appendto()的用法.分享给大家供大家参考.具体分析如下: 在jQuery的文档操作方法中,append()和appentto()方法执行的任务相同 ...

  3. css 字数超过一行显示省略号

    display:block;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;

  4. Angular.js+Bootstrap实现手风琴菜单

    说是Angular.js+Bootstrap实现手风琴菜单,其实就是用了Bootstrap的样式而已. 上一篇实现了表格+分页,接着学习实现的Demo. 主要练习自定义指令,向指令中传递参数,老规矩先 ...

  5. VCL Tclientsocket, Tserversocket控件安装方法

    菜单component->Install Packets 按Add按钮,选择delphi目录里的bin目录下的dclsockets70.bpl(delphi2010是dclsockets140. ...

  6. [原创]java WEB学习笔记44:Filter 简介,模型,创建,工作原理,相关API,过滤器的部署及映射的方式,Demo

    本博客为原创:综合 尚硅谷(http://www.atguigu.com)的系统教程(深表感谢)和 网络上的现有资源(博客,文档,图书等),资源的出处我会标明 本博客的目的:①总结自己的学习过程,相当 ...

  7. .NET: C#: System.Diagnostics

    1. Trace & Debug 理解这两者的区别,Trace有个Listners.Add()非常好用,这里网上有个在ListBox里输出Debug和Trace信息的 using System ...

  8. oracle安装过程中遇到的问题

    今天遭遇ORA-12560: TNS: 协议适配器错误的问题,经过一番努力问题已经解决,与大家共享. 造成ORA-12560: TNS: 协议适配器错误的问题的原因有三个: 1.监听服务没有起起来.w ...

  9. CSS_03_03_ul图片替换

    ul图片替换 第01步:编写css样式:url.css @charset "utf-8"; /*ul用图片替换*/ ul.u_01{/*图片*/ list-style:circle ...

  10. android中影藏状态栏和标题栏的几种方法

    1,在android中,有时候我们想隐藏我们的状态栏和标题栏(如:第一次安装app时候的欢迎界面),实现这些效果有几种方法,随便选取自己喜欢的即可. 2, A:利用代码实现,在我们主Activity中 ...