«Java.io.FileNotFoundException (нет такого файла или каталога)»

1

В моем проекте есть класс для загрузки изображения и сохранения в моей базе данных, а в случае успеха сохраните его локально. Это работает, и я могу сохранить изображение в пользовательскую папку, созданную на моем телефоне Android.

Однако в другом классе я пытаюсь загрузить закодированное изображение в текстовой форме и сохранить его локально, используя метод, который я использовал выше, но он больше не работает. Кажется, я не могу создать папку.

Я не понимаю, как это работает на моем другом классе, а не в этом

Я попытался проверить файл манифеста и посмотреть, включены ли необходимые разрешения и находятся ли они в нужных местах.

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    package="buenaventura.dorothy.tbmgmtapp">

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

    <application
        android:allowBackup="true"
        android:icon="@mipmap/lungs"
        android:label="@string/app_name"
        android:networkSecurityConfig="@xml/network_security_config"
        tools:replace="android:icon">
...

[РЕДАКТИРОВАТЬ] Хорошо, я попытался удалить пробел в моем пути к файлу, чтобы проверить, действительно ли пробел был причиной проблемы, и она все еще не работает

File folder = new File(Environment.getExternalStoragePublicDirectory( Environment.DIRECTORY_DCIM) + File.separator + "TBSystem" + File.separator + patient_id + File.separator + "lab_result_" + sqlDateToString(result_date) + ".jpg");

Кроме того, почему это работает на моем другом классе с пробелами, а не на этом? :(

File folder = new File(Environment.getExternalStoragePublicDirectory(
Environment.DIRECTORY_DCIM) + File.separator + "TB System" + File.separator + patient_id);

if (!folder.exists()) {
    folder.mkdirs();
}

folder = new File(Environment.getExternalStoragePublicDirectory(
Environment.DIRECTORY_DCIM) + File.separator + "TB System" + File.separator + patient_id + File.separator + "lab_result_" + result_date + ".jpg");

byte[] decodedBytes = Base64.decode(encoded_image, 0);
Bitmap imageBitmap = BitmapFactory.decodeByteArray(decodedBytes, 0, decodedBytes.length);

try (
    FileOutputStream out = new FileOutputStream(folder)) {
    imageBitmap.compress(Bitmap.CompressFormat.JPEG, 100, out); 
} catch (IOException e) {
    e.printStackTrace();
}

W/System.err: java.io.FileNotFoundException: /storage/emulated/0/DCIM/TB System/pat1/lab_result_2019-04-06.jpg (No such file or directory)
W/System.err:     at java.io.FileOutputStream.open0(Native Method)
        at java.io.FileOutputStream.open(FileOutputStream.java:308)
        at java.io.FileOutputStream.<init>(FileOutputStream.java:238)
        at java.io.FileOutputStream.<init>(FileOutputStream.java:180)
        at buenaventura.dorothy.tbmgmtapp.db_volley.VolleyDownloadRequest$1.onResponse(VolleyDownloadRequest.java:569)
        at buenaventura.dorothy.tbmgmtapp.db_volley.VolleyDownloadRequest$1.onResponse(VolleyDownloadRequest.java:103)
        at com.android.volley.toolbox.StringRequest.deliverResponse(StringRequest.java:82)
        at com.android.volley.toolbox.StringRequest.deliverResponse(StringRequest.java:29)
        at com.android.volley.ExecutorDelivery$ResponseDeliveryRunnable.run(ExecutorDelivery.java:102)
        at android.os.Handler.handleCallback(Handler.java:873)
        at android.os.Handler.dispatchMessage(Handler.java:99)
        at android.os.Looper.loop(Looper.java:214)
        at android.app.ActivityThread.main(ActivityThread.java:6981)
        at java.lang.reflect.Method.invoke(Native Method)
        at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:493)
W/System.err:     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1445)

Patient_id и result_date - это просто строки. encoded_image - это длинный текст

Мне нужно иметь возможность преобразовать зашифрованный текст, который я скачал, в растровое изображение и сохранить его в пользовательской папке "TB System", которую я ожидаю найти внутри DCIM. В TB System должна быть создана другая папка (то есть pat1, pat2, pat3,...), и изображения будут сохранены в них.

Теги:
file
android-permissions
android-intent

3 ответа

0

В вашем коде есть проблема, я проверил комбинации с пробелом в папке, а также с точным именем файла, на котором он работает. Фрагмент рабочего кода на основе вашей проблемы, как показано ниже,

Снимок успешно загружен

Изображение 174551

Вид компоновки кода

Изображение 174551

strings.xml

<resources>
    <string name="app_name">AlbumManager</string>
    <string name="write_button">Write Image</string>
    <string name="read_button">Read Image</string>
    <string name="request_access">Request Access</string>
</resources>

external_files.xml

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <external-path name="external_files" path="."/>
</paths>

network_security_config.xml

<?xml version="1.0" encoding="utf-8"?>
<network-security-config>
<domain-config cleartextTrafficPermitted="true">
    <domain includeSubdomains="true">maven.apache.org</domain>
    <domain includeSubdomains="true">upload.wikimedia.org</domain>
</domain-config>
</network-security-config>

GenericFileProvider.java

package com.zackdawood.albummanager;

import android.support.v4.content.FileProvider;

public class GenericFileProvider extends FileProvider {
}

FileDownloader.java

package com.zackdawood.albummanager;

import android.util.Log;

import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;

public class FileDownloader {

    private static final String TAG = "FileDownloader";

    private static final int MEGABYTE = 1024 * 1024;

    public static void downloadFile(String fileUrl, File directory) {
        try {
            Log.v(TAG, "downloadFile() invoked ");
            Log.v(TAG, "downloadFile() fileUrl " + fileUrl);
            Log.v(TAG, "downloadFile() directory " + directory);

            URL url = new URL(fileUrl);
            HttpURLConnection urlConnection = (HttpURLConnection) url.openConnection();
            urlConnection.connect();

            InputStream inputStream = urlConnection.getInputStream();
            FileOutputStream fileOutputStream = new FileOutputStream(directory);
            int totalSize = urlConnection.getContentLength();

            byte[] buffer = new byte[MEGABYTE];
            int bufferLength = 0;
            while ((bufferLength = inputStream.read(buffer)) > 0) {
                fileOutputStream.write(buffer, 0, bufferLength);
            }
            fileOutputStream.close();
            Log.v(TAG, "downloadFile() completed ");

        } catch (FileNotFoundException e) {
            e.printStackTrace();
            Log.e(TAG, "downloadFile() error" + e.getMessage());
            Log.e(TAG, "downloadFile() error" + e.getStackTrace());
        } catch (MalformedURLException e) {
            e.printStackTrace();
            Log.e(TAG, "downloadFile() error" + e.getMessage());
            Log.e(TAG, "downloadFile() error" + e.getStackTrace());
        } catch (IOException e) {
            e.printStackTrace();
            Log.e(TAG, "downloadFile() error" + e.getMessage());
            Log.e(TAG, "downloadFile() error" + e.getStackTrace());
        }
    }
}

MainActivity.java

package com.zackdawood.albummanager;

import android.content.ActivityNotFoundException;
import android.content.Context;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.net.Uri;
import android.os.AsyncTask;
import android.os.Build;
import android.os.Bundle;
import android.os.Environment;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.FileProvider;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import android.widget.Toast;

import java.io.File;
import java.io.IOException;

public class MainActivity extends AppCompatActivity {

    private static final String TAG = "MainActivity";
    private static final String[] PERMISSIONS = {android.Manifest.permission.READ_EXTERNAL_STORAGE,
            android.Manifest.permission.WRITE_EXTERNAL_STORAGE,
            android.Manifest.permission.INTERNET};

    private static boolean hasPermissions(Context context, String... permissions) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M && context != null && permissions != null) {
            for (String permission : permissions) {
                if (ActivityCompat.checkSelfPermission(context, permission) != PackageManager.PERMISSION_GRANTED) {
                    return false;
                }
            }
        }
        return true;
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ActivityCompat.requestPermissions(MainActivity.this, PERMISSIONS, 112);
        Log.v(TAG, "onCreate() Method invoked ");
    }

    public void write(View view) {
        if (!hasPermissions(MainActivity.this, PERMISSIONS)) {

            Log.v(TAG, "download() Method DON'T HAVE PERMISSIONS ");

            Toast t = Toast.makeText(getApplicationContext(), "You don't have write access !", Toast.LENGTH_LONG);
            t.show();

        } else {

            new DownloadFile().execute("https://upload.wikimedia.org/wikipedia/commons/thumb/c/c2/Adler.jpg/547px-Adler.jpg", "lab_result_2019-04-06.jpg");


        }
    }

    public void read(View view) {
        if (!hasPermissions(MainActivity.this, PERMISSIONS)) {

            Log.v(TAG, "download() Method DON'T HAVE PERMISSIONS ");

            Toast t = Toast.makeText(getApplicationContext(), "You don't have read access !", Toast.LENGTH_LONG);
            t.show();

        } else {

            File folder = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM);

            File directory = new File(folder + "/TB System");
            File imgFile = new File(directory, "lab_result_2019-04-06.jpg");

            Log.v(TAG, "view() Method imgfile " + imgFile.getAbsolutePath());

            Uri path = FileProvider.getUriForFile(this, BuildConfig.APPLICATION_ID + ".fileprovider", imgFile);


            Log.v(TAG, "view() Method path " + path);

            Intent imageIntent = new Intent(Intent.ACTION_VIEW);
            imageIntent.setDataAndType(path, "image/*");
            imageIntent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
            imageIntent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);

            try {
                startActivity(imageIntent);
            } catch (ActivityNotFoundException e) {
                Toast.makeText(MainActivity.this, "No Application available to view Image", Toast.LENGTH_SHORT).show();
            }

        }
    }

    public void request(View view) {
        ActivityCompat.requestPermissions(MainActivity.this, PERMISSIONS, 112);

    }

    private class DownloadFile extends AsyncTask<String, Void, Void> {

        @Override
        protected Void doInBackground(String... strings) {
            Log.v(TAG, "doInBackground() Method invoked ");

            String fileUrl = strings[0];   // -> http://maven.apache.org/maven-1.x/maven.pdf
            String fileName = strings[1];  // -> maven.pdf
            File folder = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM);

            File directory = new File(folder + "/TB System");
            if (!directory.exists()) {
                directory.mkdir();
                Log.v(TAG, "doInBackground() Method Directory Created ");

            }

            File imgFile = new File(directory, fileName);
            Log.v(TAG, "doInBackground() imgFile invoked " + imgFile.getAbsolutePath());
            Log.v(TAG, "doInBackground() imgFile invoked " + imgFile.getAbsoluteFile());

            try {
                imgFile.createNewFile();
                Log.v(TAG, "doInBackground() file created" + imgFile);

            } catch (IOException e) {
                e.printStackTrace();
                Log.e(TAG, "doInBackground() error" + e.getMessage());
                Log.e(TAG, "doInBackground() error" + e.getStackTrace());


            }
            FileDownloader.downloadFile(fileUrl, imgFile);
            Log.v(TAG, "doInBackground() file download completed");

            return null;
        }
    }

}

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">


    <LinearLayout
        android:layout_width="395dp"
        android:layout_height="715dp"
        android:orientation="vertical"
        tools:layout_editor_absoluteX="8dp"
        tools:layout_editor_absoluteY="8dp">

        <Button
            android:id="@+id/button2"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:text="@string/write_button"
            android:onClick="write"/>

        <Button
            android:id="@+id/button3"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:text="@string/read_button"
            android:onClick="read"/>

        <Button
            android:id="@+id/button4"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:text="@string/request_access"
            android:onClick="request"/>
    </LinearLayout>
</android.support.constraint.ConstraintLayout>

AndroidManifest.xml

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

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


    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/AppTheme"
        android:networkSecurityConfig="@xml/network_security_config">
        <provider
            android:name=".GenericFileProvider"
            android:authorities="${applicationId}.fileprovider"
            android:exported="false"
            android:grantUriPermissions="true">
            <meta-data
                android:name="android.support.FILE_PROVIDER_PATHS"
                android:resource="@xml/external_files"/>
        </provider>
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

Экран запуска приложений

Изображение 174551

Читать изображение на выходе

Изображение 174551

0

Вы проверяете права на чтение и запись?

READ_EXTERNAL_STORAGE и WRITE_EXTERNAL_STORAGE должны быть зарегистрированы программно

Использовать этот

    private boolean checkPermission(){
      if(ActivityCompat.checkSelfPermission(this, 
        Manifest.permission.READ_EXTERNAL_STORAGE) != 
        PackageManager.PERMISSION_GRANTED &&
            ActivityCompat.checkSelfPermission(this, 
        Manifest.permission.WRITE_EXTERNAL_STORAGE) != 
        PackageManager.PERMISSION_GRANTED){
        return false;
    }
    return true;
}

Если правда, делай свою работу

Если false, вызовите этот метод

private static final int MY_PERMISSION_REQUEST_CODE = 1001;

ActivityCompat.requestPermissions(this, new String[]{
                Manifest.permission.READ_EXTERNAL_STORAGE,
                Manifest.permission.WRITE_EXTERNAL_STORAGE
        }, MY_PERMISSION_REQUEST_CODE);

И добавьте этот код

@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
    switch (requestCode){
        case MY_PERMISSION_REQUEST_CODE:
            if(grantResults.length > 0 && grantResults[0] == 
             PackageManager.PERMISSION_GRANTED)
                //Do your work
            break;
    }
}
  • 0
    Могу ли я сделать это тоже внутри запроса залпа?
  • 0
    Это будет ваш проект с заставкой. Этот вызов метода на странице-заставке. Звоните только сначала играть.
0

Я вижу, что путь, который вы используете, содержит пробел между "TB" и "System", это может вызвать проблемы. Ваш другой класс также содержит пути к файлам с пробелами? Если это не так, это может быть проблемой.

Проверьте этот другой вопрос, который помогает при работе с пробелами в пути к файлам. Ответ так и не был утвержден, поэтому я не уверен, что на самом деле решил проблему, но они упоминают этот фрагмент кода:

Uri uri = Uri.parse(paths);
File file = new File(uri.getPath());

Uri.parse исправляет все проблемы с пробелами/обратной косой чертой/недопустимыми символами в путях и создает "хороший" URI.

  • 0
    Да, мой другой класс также содержит пути к файлам с пробелами тоже. Это работает, хотя.
  • 0
    Ну, вы не можете просто удалить это место - если папка называется TB System, то для вашего кода нормально выдавать исключение fileNotFound, если вы пытаетесь получить доступ к TBSystem. Поскольку вы упоминаете, что другой класс работает, возможно, опубликуете этот класс и вывод кода для этого класса, чтобы мы могли сравнить эти два класса?
Показать ещё 2 комментария

Ещё вопросы

Сообщество Overcoder
Наверх
Меню