Сбой при чтении или создании файла из внешнего хранилища, несмотря на предоставленные разрешения на чтение / запись

#java #android #file-io #android-permissions

#java #Android #файл-ввод-вывод #android-разрешения #file-io

Вопрос:

это файл манифеста :

 <?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"

...

    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

    <application
        android:requestLegacyExternalStorage="true"
        android:allowBackup="true"

...

</manifest>
  

я использовал tedpermission.
Текстовый файл с именем по дате (ex:2020_10_23.txt )
из внешнего хранилища (ex:storage/emulated/0/myDiary/2020_10_23.txt )
для операций ввода-вывода.

здесь основной код в mainactivity :

  PermissionListener permissionlistener = new PermissionListener() {
        @Override
        public void onPermissionGranted() {
            Toast.makeText(MainActivity.this, "Permission granted", Toast.LENGTH_SHORT).show();
        }

        @Override
        public void onPermissionDenied(List<String> deniedPermissions) {
            Toast.makeText(MainActivity.this, "Permission deniedn"   deniedPermissions.toString(), Toast.LENGTH_SHORT).show();
        }
    };

    @RequiresApi(api = Build.VERSION_CODES.R)
    @Override
    public void onCreate(Bundle savedInstanceState) {
        TedPermission.with(this)
                .setPermissionListener(permissionlistener)
                .setDeniedMessage("External Storage READ/WRITE Denied")
                .setPermissions(Manifest.permission.WRITE_EXTERNAL_STORAGE, Manifest.permission.READ_EXTERNAL_STORAGE)
                .check();

...

   final String sdPath = Environment.getExternalStorageDirectory().getAbsolutePath();
   final File myDir = new File(sdPath   "/myDiary");
   string fileName = Integer.toString(year)   "_"
                          Integer.toString(monthOfYear   1)   "_"
                          Integer.toString(dayOfMonth)   ".txt";

...

 btnWrite.setOnClickListener(new View.OnClickListener() {
            @RequiresApi(api = Build.VERSION_CODES.O)
            public void onClick(View v) {
                try {
                    //openFileoutput Methods does not allow path delimiters.                    
                    File current_file = new File(myDir   "/"   fileName);

                    if (!current_file.getParentFile().exists()) {
                        if (!current_file.getParentFile().mkdirs()) {
                            Toast.makeText(getApplicationContext(),
                                    "Failed to create folder : "   myDir, Toast.LENGTH_SHORT).show();
                        }
                    }

                    if (!current_file.exists()) {
                        if (!current_file.createNewFile()) {
                            Toast.makeText(getApplicationContext(),
                                    "Failed to create file : "   myDir   "/"   fileName, Toast.LENGTH_SHORT).show();
                        }
                    }

                    FileOutputStream outFs = new FileOutputStream(new File(String.valueOf(current_file)));

                    String str = edtDiary.getText().toString();
                    outFs.write(str.getBytes());
                    outFs.close();

                    Toast.makeText(getApplicationContext(),
                            fileName   " is saved", Toast.LENGTH_SHORT).show();

                } catch (IOException e) {
                    Toast.makeText(getApplicationContext(),
                            "Failed to save file : "   myDir   "/"   fileName, Toast.LENGTH_SHORT).show();
                    e.printStackTrace();
                }

            }
        });

...
//read text file from external storage
String readDiary(String fName) {
        String diaryStr = null;
        FileInputStream inFs;
        try {
            inFs = new FileInputStream(new File(fName));
            byte[] txt = new byte[500];
            inFs.read(txt);
            inFs.close();
            diaryStr = (new String(txt)).trim();
            btnWrite.setText("modify");
        } catch (IOException e) {
            edtDiary.setHint("no existing diary");
            btnWrite.setText("new save");
        }
        return diaryStr;
    }



  

Версия Android: 11

Разрешения приложения :

введите описание изображения здесь

а это logcat:

 2020-10-23 15:40:00.986 6178-6178/com.cookandroid.project8_1 W/System.err: java.io.IOException: No such file or directory
2020-10-23 15:40:00.986 6178-6178/com.cookandroid.project8_1 W/System.err:     at java.io.UnixFileSystem.createFileExclusively0(Native Method)
2020-10-23 15:40:00.986 6178-6178/com.cookandroid.project8_1 W/System.err:     at java.io.UnixFileSystem.createFileExclusively(UnixFileSystem.java:317)
2020-10-23 15:40:00.986 6178-6178/com.cookandroid.project8_1 W/System.err:     at java.io.File.createNewFile(File.java:1008)
2020-10-23 15:40:00.986 6178-6178/com.cookandroid.project8_1 W/System.err:     at com.cookandroid.project8_1.MainActivity$3.onClick(MainActivity.java:102)
2020-10-23 15:40:00.987 6178-6178/com.cookandroid.project8_1 W/System.err:     at android.view.View.performClick(View.java:7448)
2020-10-23 15:40:00.987 6178-6178/com.cookandroid.project8_1 W/System.err:     at android.view.View.performClickInternal(View.java:7425)
2020-10-23 15:40:00.987 6178-6178/com.cookandroid.project8_1 W/System.err:     at android.view.View.access$3600(View.java:810)
2020-10-23 15:40:00.987 6178-6178/com.cookandroid.project8_1 W/System.err:     at android.view.View$PerformClick.run(View.java:28305)
2020-10-23 15:40:00.987 6178-6178/com.cookandroid.project8_1 W/System.err:     at android.os.Handler.handleCallback(Handler.java:938)
2020-10-23 15:40:00.987 6178-6178/com.cookandroid.project8_1 W/System.err:     at android.os.Handler.dispatchMessage(Handler.java:99)
2020-10-23 15:40:00.987 6178-6178/com.cookandroid.project8_1 W/System.err:     at android.os.Looper.loop(Looper.java:223)
2020-10-23 15:40:00.987 6178-6178/com.cookandroid.project8_1 W/System.err:     at android.app.ActivityThread.main(ActivityThread.java:7656)
2020-10-23 15:40:00.987 6178-6178/com.cookandroid.project8_1 W/System.err:     at java.lang.reflect.Method.invoke(Native Method)
2020-10-23 15:40:00.987 6178-6178/com.cookandroid.project8_1 W/System.err:     at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:592)
2020-10-23 15:40:00.987 6178-6178/com.cookandroid.project8_1 W/System.err:     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:947)
  

он не может создать папку ‘myDiary’ в хранилище / emulated / 0
Я перепробовал все, что мог. почему невозможно создать папку, файл, даже если мне предоставлены разрешения ввода-вывода внешнего хранилища?

Ответ №1:

На Android 11 у вас нет доступа к корню внешнего хранилища.

Вы не можете создавать файлы или папки в /storage/emulated/0 .

Но .. вы можете записывать в обычные общедоступные папки в /storage /emulated/ 0, например

 Download, Documents, DCIM, Pictures, Alarms, a.s.o.
  

Посмотрите, какие папки уже есть на вашем устройстве, и вы можете в основном использовать их для операций чтения и / или записи.

Комментарии:

1. Могу ли я получить доступ к корню внешнего хранилища (storage / emulated / 0) с помощью Mediastore или SAF (платформа доступа к хранилищу)?

2. Он должен быть записан в хранилище / эмулированный / 0 / myDiary.

3. С SAF да. Вы можете позволить пользователю создать такую папку с помощью ACTION_OPEN_DOCUMENT_TREE, а затем выбрать ее. После этого ваше приложение может использовать эту папку. С MediaStore нет.

4. Но даже если SAF сможет создать myDiary, у вас не будет доступа по пути /storage/emulated/0/myDiary, но вы должны использовать полученный saf uri -content scheme — для этой папки. Но… пожалуйста, просто попробуйте.

5. It must be recorded in ... Почему?

Ответ №2:

Используйте это

 public class Folder
{
    private final File appDataFolder;

    public Folder()
    {
        File file = Environment.getExternalStorageDirectory();

        appDataFolder = new File(file, "AppDataFolder");

        if (!appDataFolder.exists()) appDataFolder.mkdir();
    }

    public String getMainFolder()
    {
        return appDataFolder.getAbsolutePath();
    }

    public void deleteMainFolder()
    {
        if (appDataFolder.isDirectory())
        {
            String[] children = appDataFolder.list();

            for (String child : children)
            {
                new File(appDataFolder, child).delete();
            }
        }

        new Folder();
    }

    public void createNewFolder(String folderName)
    {
        File file = new File(appDataFolder, folderName);

        if (!file.exists()) file.mkdir();
    }

    public void createChildFolder(String parentName, String childName)
    {
        String path = appDataFolder.getAbsolutePath()   File.separator   parentName;

        File parent = new File(path);

        File file = new File(parent, childName);

        if (!file.exists()) file.mkdir();
    }
}
  

и в деятельности

 PermissionListener permissionlistener = new PermissionListener() {
        @Override
        public void onPermissionGranted() {
            new Folder().createNewFolder("myDiary");
            Toast.makeText(MainActivity.this, "Permission granted", Toast.LENGTH_SHORT).show();
        }

        @Override
        public void onPermissionDenied(List<String> deniedPermissions) {
            Toast.makeText(MainActivity.this, "Permission deniedn"   deniedPermissions.toString(), Toast.LENGTH_SHORT).show();
        }
    };
  

Комментарии:

1. пробовал это, но та же проблема: java.io.IOException: нет такого файла или каталога

2. он не может создать папку «myDiary»

3. проверьте редактирование. попробуйте создать новую папку (); после предоставления всех разрешений

4. ах .. та же проблема, что не создает папку. Проводник файлов устройства: imgur.com/w20nAVV

5. «java.io.IOException: нет такого файла или каталога» произошло, потому что вы пытаетесь получить доступ к каталогу, который даже не создан.