【问题标题】:App crashes when picking image from external storage on android 5.1.1在 android 5.1.1 上从外部存储中选择图像时应用程序崩溃
【发布时间】:2015-12-29 04:27:25
【问题描述】:

在我的应用中,用户可以从图库中选择图片:

private void pickPicture() {
    Intent intent = new Intent(Intent.ACTION_PICK, android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI);

    this.startActivityForResult(intent, Globals.REQUEST_PICK_PHOTO);
}

通过上面的代码,我的画廊打开了,它显示了我所有的照片,但它绝对不是我手机上的旧画廊(我使用的是最新的 android 版本 5.1.1 的 google nexus 4)。新画廊如下所示:

但是,当我现在单击图像时,应用程序崩溃了:

07-29 18:30:41.896    2693-2693/de.mypackage.myproject E/CursorWindow﹕ Failed to read row 0, column -1 from a CursorWindow which has 1 rows, 7 columns.
07-29 18:30:41.896    2693-2693/de.mypackage.myproject D/AndroidRuntime﹕ Shutting down VM
07-29 18:30:41.898    2693-2693/de.mypackage.myproject E/AndroidRuntime﹕ FATAL EXCEPTION: main
    Process: de.mypackage.myproject, PID: 2693
    java.lang.RuntimeException: Failure delivering result ResultInfo{who=null, request=65538, result=-1, data=Intent { dat=content://com.google.android.apps.photos.contentprovider/0/1/content://media/external/images/media/10961/ACTUAL/102077033 flg=0x1 (has clip) }} to activity {de.mypackage.myproject/de.mypackage.ProfileEditPictureActivity}: java.lang.IllegalStateException: Couldn't read row 0, col -1 from CursorWindow.  Make sure the Cursor is initialized correctly before accessing data from it.
            at android.app.ActivityThread.deliverResults(ActivityThread.java:3574)
            at android.app.ActivityThread.handleSendResult(ActivityThread.java:3617)
            at android.app.ActivityThread.access$1300(ActivityThread.java:151)
            at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1352)
            at android.os.Handler.dispatchMessage(Handler.java:102)
            at android.os.Looper.loop(Looper.java:135)
            at android.app.ActivityThread.main(ActivityThread.java:5254)
            at java.lang.reflect.Method.invoke(Native Method)
            at java.lang.reflect.Method.invoke(Method.java:372)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:903)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:698)
     Caused by: java.lang.IllegalStateException: Couldn't read row 0, col -1 from CursorWindow.  Make sure the Cursor is initialized correctly before accessing data from it.
            at android.database.CursorWindow.nativeGetString(Native Method)
            at android.database.CursorWindow.getString(CursorWindow.java:438)
            at android.database.AbstractWindowedCursor.getString(AbstractWindowedCursor.java:51)
            at android.database.CursorWrapper.getString(CursorWrapper.java:114)
            at de.mypackage.utils.Helper.getRealPathFromURI(Helper.java:1064)
            at de.mypackage.fragments.profile.picture.ProfilePictureEditFragment.onActivityResult(ProfilePictureEditFragment.java:109)
            at android.support.v4.app.FragmentActivity.onActivityResult(FragmentActivity.java:165)
            at de.mypackage.activities.ProfileEditPictureActivity.onActivityResult(ProfileEditPictureActivity.java:51)
            at android.app.Activity.dispatchActivityResult(Activity.java:6192)
            at android.app.ActivityThread.deliverResults(ActivityThread.java:3570)
            at android.app.ActivityThread.handleSendResult(ActivityThread.java:3617)
            at android.app.ActivityThread.access$1300(ActivityThread.java:151)
            at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1352)
            at android.os.Handler.dispatchMessage(Handler.java:102)
            at android.os.Looper.loop(Looper.java:135)
            at android.app.ActivityThread.main(ActivityThread.java:5254)
            at java.lang.reflect.Method.invoke(Native Method)
            at java.lang.reflect.Method.invoke(Method.java:372)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:903)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:698)

我的onActivityResult

@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);

    if(resultCode == Activity.RESULT_OK) {
        if(requestCode == Globals.REQUEST_TAKE_PHOTO) {
            this.startCropImage();
        } else if (requestCode == Globals.REQUEST_PICK_PHOTO) {
            this.imageFile = new File(Helper.getRealPathFromURI(this.getActivity(), data.getData()));

            this.startCropImage();
        } else if (requestCode == Globals.REQUEST_CROP_PHOTO) {
            this.imageView.setImageBitmap(Images.loadBitmapFromFile(Paths.IMAGE_PROFILE));

            String message = this.getResources().getString(R.string.text_profile_save_completed);

            Toast.makeText(this.getActivity(), message, Toast.LENGTH_LONG).show();
        }
    }
}

现在是应用崩溃的代码:

public static String getRealPathFromURI(Context context, Uri contentURI) {
    String result = null;
    Cursor cursor = context.getContentResolver().query(contentURI, null, null, null, null);

    if (cursor == null) {
        result = contentURI.getPath();
    } else { 
        cursor.moveToFirst(); 

        int index = cursor.getColumnIndex(MediaStore.Images.ImageColumns.DATA); 

        result = cursor.getString(index);
        cursor.close();
    }

    return result;
}

应用程序崩溃的确切时间:

result = cursor.getString(index)

我上面发布的代码适用于所有早于 5.1.1 的 android 版本,但不适用于 5.1.1,因此需要有所不同。你知道问题出在哪里吗?

【问题讨论】:

    标签: android


    【解决方案1】:

    我上面发布的代码适用于所有早于 5.1.1 的 android 版本

    仅在您尝试过的几台设备上,并且仅适用于您尝试过的照片。

    A Uri is not a file。您无法获得图像的“真实路径”,因为不要求 Uri 指向您可以访问的文件。例如,您的代码不适用于超过 4 亿台设备(所有设备均搭载 Android 4.4+),因为您没有对可移动媒体上任意位置的读取权限。

    使用UriContentResolver 来获取由Uri 和MIME 类型表示的字节,就像使用HttpUrlConnection 来获取URL 的字节和MIME 类型一样。

    【讨论】:

    • 所以我猜 Paul Bruke 的答案在这里:stackoverflow.com/questions/20067508/… 会解决它。
    • @Mulgard:不会。例如,对于可移动媒体上的照片,您的应用仍然会在超过 4 亿台设备(Android 4.4+ 附带的所有设备)上被破坏,因为您没有阅读访问可移动媒体上的任意位置。 “解决”的方法就是按照我写的做,用ContentResolver访问Uri标识的内容。
    • 我以为就是这样。我在网络上找不到任何其他使用 ContentResolver 的解决方案。
    • 哦,就这么简单吗? getContentResolver().openInputStream(uri) 但这意味着我需要为每个图像创建一个临时文件,因为我只使用文件。
    • @Mulgard:理想情况下,您可以修复其余代码,使其不适用于“仅文件”。
    【解决方案2】:

    当您没有获取此类列时会发生此错误。 为了确保只打印出你的光标

    System.out.println(DatabaseUtils.dumpCursorToString(yourcursor));
    

    看看有没有这样的专栏。

    【讨论】:

      【解决方案3】:

      试试这个方法,看看这个question

      public String getImagePath(Uri uri){
         Cursor cursor = getContentResolver().query(uri, null, null, null, null);
         cursor.moveToFirst();
         String document_id = cursor.getString(0);
         document_id = document_id.substring(document_id.lastIndexOf(":")+1);
         cursor.close();
      
         cursor = getContentResolver().query( 
         android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
         null, MediaStore.Images.Media._ID + " = ? ", new String[]{document_id}, null);
         cursor.moveToFirst();
         String path = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA));
         cursor.close();
      
         return path;
      }
      

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 2016-07-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        相关资源
        最近更新 更多