Android:获取任何文件的文件名和扩展名

Mar*_*ang -2 android android-emulator

我遇到的问题是,我无法从搭载 Android 9 的智能手机上选择的文件中提取文件扩展名。该问题不会出现在搭载 Android 8 或更低版本的设备上。

当用户选择任何类型的文件时,onActivityResult 被调用。在这个方法中,我从 FileUtils 类调用 getPath,所以我可以提取文件扩展名。

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    Uri uri = new Uri.Builder().build();
    if (resultCode == RESULT_OK) {
        switch (requestCode) {
            case PICKFILE_REQUEST_CODE: {
                String path = new File(data.getData().getPath()).getAbsolutePath();

                if (path != null) {
                    uri = data.getData();
                    path = FileUtils.getPath(getApplicationContext().getContentResolver(), uri);
                    name = path.substring(path.lastIndexOf("."));
                    extension = path.substring(path.lastIndexOf(".") + 1);
                }
                break;
            }
        }
    }
Run Code Online (Sandbox Code Playgroud)

FileUtils类(来自此 repo)的getPath()通过 URI 评估文件是哪种文件,但对这种情况负责的是以下部分:

public static String getPath(final ContentResolver cr, final Uri uri) {
    if (isDownloadsDocument(uri)) {
        final String id = DocumentsContract.getDocumentId(uri);

        final List<Uri> uris = new ArrayList<>();
        uris.add(ContentUris.withAppendedId(
                 Uri.parse("content://downloads/public_downloads"), Long.valueOf(id)));
        uris.add(ContentUris.withAppendedId(
                 Uri.parse("content://downloads/my_downloads"), Long.valueOf(id)));
        uris.add(ContentUris.withAppendedId(
                 Uri.parse("content://downloads/all_downloads"), Long.valueOf(id)));

        String res;
        for (int i = 0; i < uris.size(); i++) {
            res = getDataColumn(cr, uris.get(i), null, null);
            if (res != null) return res;
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

getDataColumn()查询保存文件物理路径的文件列:

public static String getDataColumn(ContentResolver cr,
                                   Uri uri, String selection,
                                   String[] selectionArgs) {
    Cursor cursor = null;
    final String[] projection = {"_data"};
    String col = "";

    try {
        cursor = cr.query(uri, projection, selection, selectionArgs,null);
        if (cursor != null && cursor.moveToFirst()) {
            final int column_index = cursor.getColumnIndexOrThrow("_data");
            col = cursor.getString(column_index);
        }
    }finally{ if (cursor != null) cursor.close();}
    return col;
}
Run Code Online (Sandbox Code Playgroud)

当我测试它的路径,我得到了看起来像例如:E/AddItemActivity: result, path: /storage/emulated/0/Download/SampleVideo_176x144_1mb.3gp。但是对于 Android 9,路径始终为空。

Mar*_*ang 6

对于解决方案,我使用 MediaStore 来提取所选文件的名称和扩展名。所以我onActivityResult用以下代码替换代码:

String path = new File(data.getData().getPath()).getAbsolutePath();

if(path != null){
  uri = data.getData();

  String filename;
  Cursor cursor = getContentResolver().query(uri,null,null,null,null);
  
  if(cursor == null) filename=uri.getPath();
  else{
    cursor.moveToFirst();
    int idx = cursor.getColumnIndex(MediaStore.Files.FileColumns.DISPLAY_NAME);
    filename = cursor.getString(idx);
    cursor.close();
  }

  String name = filename.substring(0,filename.lastIndexOf("."));
  String extension = filename.substring(filename.lastIndexOf(".")+1);
}
Run Code Online (Sandbox Code Playgroud)

因此,MediaStore.File.FileColumns.DISPLAY_NAME可以从使用文件选择器选择的任何文件中获取文件名和扩展名。