【问题标题】:ExifInterface constructor throws IOExxceptionExifInterface 构造函数抛出 IOExxception
【发布时间】:2016-10-16 01:55:54
【问题描述】:

在尝试使用文件路径初始化 exif 接口实例时,我无法确定 ExifInterface 构造函数引发的异常。

更新 请按要求查看下面的详细代码。

文件下载功能

public void downloadAndSaveFile(String url, String directoryId, String fileName) {
    HttpURLConnection conn = null;
    try {
        Log.d(TAG, "DownloadFileTask url : " + url);
        conn = getGETConnection(url);
        conn.setRequestProperty("Accept", "application/json");
        conn.setRequestProperty("Content-Type", "application/json");
        conn.setRequestProperty("Authorization", "Bearer " + MY_AUTH_TOKEN);
        conn.connect();
        File file = new File(FileTools.getCacheFileLocation(fileName, directoryId));
        FileOutputStream fileOutput = new FileOutputStream(file);
        InputStream inputStream = (InputStream) conn.getInputStream();
        byte[] buffer = new byte[1024 * 1024];
        int bufferLength = 0;
        while ((bufferLength = inputStream.read(buffer)) > 0) {
            fileOutput.write(buffer, 0, bufferLength);
        }
        fileOutput.close();
        inputStream.close();
        Log.d(TAG, "Download successful. Downloaded File : " + file.getAbsolutePath());
        // Generate thumbnail and encrypt the file and thumbnail
        ***String thumbnailPath = FileTools.cacheThumbnail(file, file.getName(), directoryId);***
        if (thumbnailPath != null && !thumbnailPath.isEmpty()) {
            // Encrypt the file
            try {
                FileTools.SaveFileEncrypted(file, directoryId);
            } catch (Exception e) {
                e.printStackTrace();
                // Delete the file
                file.delete();
            }
        } else {
            // Delete the file
            file.delete();
        }
    } catch (Exception e) {
        Log.d(TAG, "Exception while downloading file");
        e.printStackTrace();
    } finally {
        if (conn != null) {
            conn.disconnect();
        }
    }
}

FileTools.cacheThumbnail 函数

public static String cacheThumbnail(File file, String fileName, String chatId) {
    Log.d(TAG, "cacheThumbnail original File path : " + file.getAbsolutePath());
    Log.d(TAG, "cacheThumbnail original File exists : " + file.exists());
    Log.d(TAG, "cacheThumbnail original file size : " + file.length());
    String thumbName = String.format("thumb-%s.jpg", fileName);
    File fileThumb = new File(FileTools.getMediaCachePath(chatId), thumbName);
    ByteArrayOutputStream out = null;
    try {
        out = new ByteArrayOutputStream();
        BitmapFactory.Options options = new BitmapFactory.Options();
        Bitmap image = ThumbnailUtils.extractThumbnail(BitmapFactory.decodeFile(file.toURI().getPath(), options), 256, 256, ThumbnailUtils.OPTIONS_RECYCLE_INPUT);
        // Thumbnail generation for the image failed. It is a video file. Generate
        // thumbnail for the video file
        if (image == null) {
            MediaMetadataRetriever retriever = new MediaMetadataRetriever();
            try {
                retriever.setDataSource(FileTools.getFileInputStreamFromStorage(file).getFD());
                // Generate thumbnail from a frame that is 5% deep into the video
                String time = retriever.extractMetadata(MediaMetadataRetriever.METADATA_KEY_DURATION);
                long timeInMillisec = Long.parseLong(time);
                long durationMicroSec = timeInMillisec * 1000;
                long thumbnailDepth = (long) (durationMicroSec * (0.15f));
                image = retriever.getFrameAtTime(thumbnailDepth, MediaMetadataRetriever.OPTION_CLOSEST);
                image = ThumbnailUtils.extractThumbnail(image, 256, 256, ThumbnailUtils.OPTIONS_RECYCLE_INPUT);
            } catch (IllegalArgumentException ex) {
                Log.e(TAG, ex.getMessage());
                image = null;
            } catch (RuntimeException ex) {
                Log.e(TAG, ex.getMessage());
                image = null;
            } catch (IOException e) {
                image = null;
                Log.e(TAG, e.getMessage());
            }
        }
        // Could not generate a thumbnail. Probably a corrupt/bad file
        if (image == null) {
            return null;
        }
        ***image = Utilities.orientBitmap(file.getAbsolutePath(), image);***
        image.compress(Bitmap.CompressFormat.JPEG, 80, out);
        out.close();
        FileTools.SaveFileEncrypted(fileThumb, out.toByteArray());
        return fileThumb.toURI().getPath();
    } catch (Exception e) {
        e.printStackTrace();
        return null;
    } finally {
        try {
            if (out != null) {
                out.close();
            }
        } catch (Exception ignore) {
        }
    }
}

orientBitmap和rotateBitmap函数函数

public static Bitmap orientBitmap(String filePath, Bitmap bitmap) throws IOException {
    Log.d(TAG, "orientBitmap FilePath : " + filePath);
    File file = new File(filePath);
    Log.d(TAG, "orientBitmap File exists : " + file.exists());

    ExifInterface exifInterface = new ExifInterface(filePath);
    Log.d(TAG, "After exception");
    int orientation = exifInterface.getAttributeInt(ExifInterface.TAG_ORIENTATION, ExifInterface.ORIENTATION_NORMAL);

    Log.d(TAG, "Thumb orientation : " + orientation);

    switch (orientation) {
        case ExifInterface.ORIENTATION_ROTATE_90:
            bitmap = rotateBitmap(bitmap, 90);
            break;
        case ExifInterface.ORIENTATION_ROTATE_180:
            bitmap = rotateBitmap(bitmap, 180);
            break;
        case ExifInterface.ORIENTATION_ROTATE_270:
            bitmap = rotateBitmap(bitmap, 270);
            break;
    }
    return bitmap;
}

public static Bitmap rotateBitmap(Bitmap source, float angle) {
    Matrix matrix = new Matrix();
    matrix.postRotate(angle);
    return Bitmap.createBitmap(source, 0, 0, source.getWidth(), source.getHeight(), matrix, true);
}

在尝试生成 mp4 文件和 png 文件的缩略图时,我通过发布堆栈跟踪来更新堆栈跟踪。

mp4 文件的堆栈跟踪。

D/DownloadFileClass: downloadAndSaveFile url : example.com/media/download/568405
D/DownloadFileClass: Download successful. Downloaded File : /data/user/0/com.mypackage.myapp/cache/userhash/79640/media/MP4_20161017_134641.mp4
D/FileTools: cacheThumbnail original File path : /data/user/0/com.mypackage.myapp/cache/userhash/79640/media/MP4_20161017_134641.mp4
D/FileTools: cacheThumbnail original File exists : true
D/FileTools: cacheThumbnail original file size : 6434816
D/skia: --- SkImageDecoder::Factory returned null
D/Utilities: orientBitmap FilePath : /data/user/0/com.mypackage.myapp/cache/userhash/79640/media/MP4_20161017_134641.mp4
D/Utilities: orientBitmap File exists : true
W/ExifInterface: Invalid image.
    java.io.IOException: Invalid marker: 0
    at android.media.ExifInterface.getJpegAttributes(ExifInterface.java:1600)
           at android.media.ExifInterface.loadAttributes(ExifInterface.java:1339)
           at android.media.ExifInterface.<init>(ExifInterface.java:1057)
           at com.mypackage.helpers.Utilities.orientBitmap(Utilities.java:85)
           at com.mypackage.fileio.FileTools.cacheThumbnail(FileTools.java:700)
           at com.mypackage.coreapi.DownloadFileClass$downloadAndSaveFile(DownloadFileClass.java:215)
           at com.mypackage.coreapi.DownloadFileClass$downloadAndSaveFile(DownloadFileClass.java:113)
           at android.os.AsyncTask$2.call(AsyncTask.java:295)
           at java.util.concurrent.FutureTask.run(FutureTask.java:237)
           at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:234)
           at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1113)
           at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:588)
           at java.lang.Thread.run(Thread.java:818)
D/Utilities: After exception
D/Utilities: Thumb orientation : 0
D/FileTools: Save file to : /data/user/0/com.mypackage.myapp/cache/userhash/79640/media/MP4_20161017_134641.mp4
D/EncryptedFileTools: ......Final saved file path...... : /data/user/0/com.mypackage.myapp/cache/userhash/79640/media/encryptPlaceHolder

png 文件的堆栈跟踪

D/DownloadFileClass: downloadAndSaveFile url : example.com/media/download/568406
D/DownloadFileClass: Download successful. Downloaded File : /data/user/0/com.mypackage.myapp/cache/userhash/79640/media/PNG_20161017_134748.png
D/FileTools: cacheThumbnail original File path : /data/user/0/com.mypackage.myapp/cache/userhash/79640/media/PNG_20161017_134748.png
D/FileTools: cacheThumbnail original File exists : true
D/FileTools: cacheThumbnail original file size : 92160
D/Utilities: orientBitmap FilePath : /data/user/0/com.mypackage.myapp/cache/userhash/79640/media/PNG_20161017_134748.png
D/Utilities: orientBitmap File exists : true
W/ExifInterface: Invalid image.
    java.io.IOException: Invalid marker: 89
           at android.media.ExifInterface.getJpegAttributes(ExifInterface.java:1600)
           at android.media.ExifInterface.loadAttributes(ExifInterface.java:1339)
           at android.media.ExifInterface.<init>(ExifInterface.java:1057)
           at com.mypackage.helpers.Utilities.orientBitmap(Utilities.java:85)
           at com.mypackage.fileio.FileTools.cacheThumbnail(FileTools.java:700)
           at com.mypackage.coreapi.DownloadFileClass$downloadAndSaveFile(DownloadFileClass.java:215)
           at com.mypackage.coreapi.DownloadFileClass$downloadAndSaveFile(DownloadFileClass.java:113)
           at android.os.AsyncTask$2.call(AsyncTask.java:295)
           at java.util.concurrent.FutureTask.run(FutureTask.java:237)
           at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:234)
           at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1113)
           at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:588)
           at java.lang.Thread.run(Thread.java:818)
D/Utilities: After exception
D/Utilities: Thumb orientation : 0
D/FileTools: Save file to : /data/user/0/com.mypackage.myapp/cache/userhash/79640/media/PNG_20161017_134748.png
D/EncryptedFileTools: ......Final saved file path...... : /data/user/0/com.mypackage.myapp/cache/userhash/79640/media/encryptPlaceHolder

令人困惑的部分是,当我在装有 android 4.4.4 的三星 Galaxy S4 上运行该应用程序时,我没有收到此异常,但当我在装有 android 6.0.1 的三星 Galaxy S5 上运行它时,我总是得到它。我也在使用 Android 7.0 的模拟器上收到此错误。我不确定如何解决这个问题。

请查看随附的权限屏幕截图。 Enabled Permissions

我已添加要求的信息。如果需要更多信息,请告诉我。我之前没有使用过 ExifTags,所以请多多包涵。 再次感谢。

【问题讨论】:

  • 您有权访问该文件吗?如果您面向 API 23 或更高版本,Android 6.0 要求您在运行时请求外部存储权限
  • @ianhanniballake 感谢您的回复。我已启用所有权限。我添加了权限页面的屏幕截图供您参考。如果您需要更多详细信息,请告诉我。
  • 你传递给方法的路径是什么?还要确保文件存在
  • 我的猜测是filePath 不是一个实际的文件系统路径。请显示您获取此值的代码。
  • 这主要是一个很好的问题。我要求的一项调整是,如果您的日志向右缩进很多,请将它们向左移动,这样读者就不需要滚动来阅读它们,谢谢。

标签: android image-rotation


【解决方案1】:

我在上面的代码中发现了问题。显然我试图直接从 mp4 和 png 文件中读取 EXIF 标签。在搜索有关 EXIF 标签的更多信息后,我发现 EXIF 标签仅适用于 JPEG 文件。所以我的问题的解决方法是首先创建一个 mp4/png 文件的 JPEG 图像,然后尝试从这个 JPEF 文件中读取 EXIF 标签。这样做解决了问题,我不再收到异常。

感谢大家指导我解决问题。

【讨论】:

  • 你是如何解决这个问题的,你能解释一下吗?
  • @praneethkumar 正如我上面提到的,我试图直接从 mp4/png 文件中读取 EXIF 标签。由于 EXIF 标签仅与 JPEG 文件相关联,因此当我尝试从 mp4/png 文件中读取它们时出现此错误。我首先从 mp4/png 文件创建 JPEG,然后尝试从 JPEG 文件中读取 EXIF 标签,从而防止了该错误。这种方法可以防止错误,但我不相信您会从 mp4/png 创建的 JPEG 文件中获得任何 EXIF 信息,除非您在从 mp4/png 文件创建后明确设置 JPEG 文件的 EXIF 值。
猜你喜欢
  • 2011-06-02
  • 1970-01-01
  • 1970-01-01
  • 2011-04-07
  • 2014-08-10
  • 2011-11-04
  • 1970-01-01
  • 2013-01-01
  • 2015-09-10
相关资源
最近更新 更多