Android中讀取圖片EXIF元數據之metadata-extractor的使用



一、引言及介紹


近期在開發中用到了metadata-extractor-xxx.jar 和 xmpcore-xxx.jar這個玩意, 索性查閱大量文章了解學習,來分享分享。

本身工作也是常常和處理大圖片打交道,摸索摸索也是多多益善。

首先介紹一下什么是EXIF。EXIF是 Exchangeable Image File 的縮寫,這是一種專門為數碼相機照片設定的格式。這樣的格式能夠用來記錄數字照片的屬性信息,如相機的品牌及型號、相片的拍攝時間、拍攝時所設置的光圈大小、快門速度、ISO等信息。除此之外它還能夠記錄拍攝數據,以及圖片格式化方式。這樣就能夠輸出到兼容EXIF格式的外設上。如照片打印機等。


眼下最常見的支持EXIF信息的圖片格式是JPG。非常多的圖像工具都能夠直接顯示圖片的EXIF信息,包括如今的一些著名的相冊站點也提供頁面用於顯示照片的EXIF信息。本文介紹Java怎樣讀取圖像的EXIF信息,包括怎樣依據EXIF信息對圖像進行調整以便適合用戶瀏覽。

用BufferedImage類來讀的時候。過大的圖片時常會拋出OutOfMemoryException異常,挺蛋疼的。

BufferedImage image = ImageIO.read(File file);


眼下最簡單易用的EXIF信息處理的Java包是 Drew Noakes 寫的 metadata-extractor

是一個能夠從圖像文件里讀取元數據(Exif, IPTC, XMP, ICC等)的簡單的Java庫,使用簡單:

Metadata metadata = ImageMetadataReader.readMetadata(imagePath);

該庫能了解多種格式的元數據,當中很多能夠存在於單個圖像:
Exif、IPTC、XMP、JFIF / JFXX、ICC Profiles、Photoshop fields、PNG properties、BMP properties、GIF properties

它能處理類型的文件:JPEG、TIFF、PSD、PNG、BMP、GIF、Camera Raw (NEF/CR2/ORF/ARW/RW2/...)


注:並非每一個JPG圖像文件都包括有EXIF信息,你能夠在Windows資源管理器單擊選中圖片后。假設該圖片包括EXIF信息,則在窗體狀態欄會顯示出相機的型號。




二、演示樣例代碼及描寫敘述

以下我們給出一些代碼將含有EXIF的圖片信息所有打印出來。

演示樣例1):

import java.io.File;
    import java.util.Iterator;
     
    import com.drew.imaging.jpeg.JpegMetadataReader;
    import com.drew.metadata.Directory;
    import com.drew.metadata.Metadata;
    import com.drew.metadata.Tag;
    import com.drew.metadata.exif.ExifDirectory;
     
    /**
     * 讀取圖片的EXIF信息
     */
    public class ExifTest {
        
         public static void main(String[] args) throws Exception {
             
             //包括EXIF信息的圖片地址
             File jpegFile = new File("D:\\XXXX\\XXXX\\XXXX.JPG");
             
             Metadata metadata = JpegMetadataReader.readMetadata(jpegFile);
             
             Directory exif = metadata.getDirectory(ExifDirectory.class);
             
             Iterator tags = exif.getTagIterator();
             
             while (tags.hasNext()) {
                 
                 Tag tag = (Tag)tags.next();
                 
                 System.out.println(tag);
                 
             }
         }
    }


演示樣例2:)

public static void main(String[] args) throws Exception {

        File mFile = new File("F:/XXX.JPG");
        Metadata metadata = ImageMetadataReader.readMetadata(mFile);
        for (Directory directory : metadata.getDirectories()) {

            if("ExifSubIFDDirectory".equalsIgnoreCase( directory.getClass().getSimpleName() )){

                //光圈F值=鏡頭的焦距/鏡頭光圈的直徑
                System.out.println("光圈值: f/" + directory.getString(ExifSubIFDDirectory.TAG_FNUMBER) );

                System.out.println("曝光時間: " + directory.getString(ExifSubIFDDirectory.TAG_EXPOSURE_TIME)+ "秒" );
                System.out.println("ISO速度: " + directory.getString(ExifSubIFDDirectory.TAG_ISO_EQUIVALENT) );
                System.out.println("焦距: " + directory.getString(ExifSubIFDDirectory.TAG_FOCAL_LENGTH) + "毫米" );

                System.out.println("拍照時間: " + directory.getString(ExifSubIFDDirectory.TAG_DATETIME_ORIGINAL) );
                System.out.println("寬: " + directory.getString(ExifSubIFDDirectory.TAG_EXIF_IMAGE_WIDTH) );
                System.out.println("高: " + directory.getString(ExifSubIFDDirectory.TAG_EXIF_IMAGE_HEIGHT) );

            }

            if("ExifIFD0Directory".equalsIgnoreCase( directory.getClass().getSimpleName() )){
                System.out.println("照相機制造商: " + directory.getString(ExifIFD0Directory.TAG_MAKE) );
                System.out.println("照相機型號: " + directory.getString(ExifIFD0Directory.TAG_MODEL) );
                System.out.println("水平分辨率: " + directory.getString(ExifIFD0Directory.TAG_X_RESOLUTION) );
                System.out.println("垂直分辨率: " + directory.getString(ExifIFD0Directory.TAG_Y_RESOLUTION) );
            }
        }
    }


演示樣例3):

File mFilePath="C://XXX.jpg";

    Metadata metadata = com.drew.imaging.jpeg.JpegMetadataReader.readMetadata(mFilePath);

    JpegDirectory jd = (JpegDirectory)metadata.getDirectory(JpegDirectory.class);
    System.out.println("------------" + jd.getImageHeight()); //圖片的高
    System.out.println("------------" + jd.getImageWidth());  //圖片的寬
    
    //因為僅僅是讀取圖片的頭信息,所以不管多大的圖片都能讀取,並且速度非常快.


從運行的中能夠看到照片的具體拍攝時間。拍攝用的相機型號,曝光時間,光圈值。焦距,ISO值 等等。


你也能夠直接指定讀取當中隨意參數的值。ExifDirectory 類中定義了非常多以 TAG_ 開頭的整數常量。這些常量代表特定的一個參數值,比如要讀取相機的型號,能夠用以下代碼來獲取。


Metadata metadata = JpegMetadataReader.readMetadata(jpegFile);
        
Directory exif = metadata.getDirectory(ExifDirectory.class);
        
String model = exif.getString(ExifDirectory.TAG_MODEL);


上述提到的是怎樣獲取照片的EXIF信息,當中包括一個非常重要的信息就是——拍攝方向。

比如所用的圖片拍攝方向是:Orientation - Top, left side (Horizontal / normal)。我們在拍照的時候常常會依據場景的不同來選擇相機的方向,比如拍攝一顆高樹。我們會把相機豎着拍攝,使景物剛好適合整個取景框。可是這樣得到的圖片假設用普通的圖片瀏覽器看便是倒着的,須要調整角度才干得到一個正常的圖像。


通過讀取圖片的EXIF信息,能夠得到關於拍攝方向的這樣一個結果:Orientation - Left side, bottom (Rotate 270 CW)

而直接讀取 ExitDirectory.TAG_ORIENTATION 標簽的值是8。

來看下這個項目是怎樣來定義這些返回值的,打開源代碼包中的ExifDescriptor類的getOrientationDescription(),該方法代碼例如以下:

public String getOrientationDescription() throws MetadataException{
        
        if (!_directory.containsTag(ExifDirectory.TAG_ORIENTATION)) return null;
        
        int orientation = _directory.getInt(ExifDirectory.TAG_ORIENTATION);
        
        switch (orientation) {
            
            case 1: return "Top, left side (Horizontal / normal)";
            case 2: return "Top, right side (Mirror horizontal)";
            case 3: return "Bottom, right side (Rotate 180)";
            case 4: return "Bottom, left side (Mirror vertical)";
            case 5: return "Left side, top (Mirror horizontal and rotate 270 CW)";
            case 6: return "Right side, top (Rotate 90 CW)";
            case 7: return "Right side, bottom (Mirror horizontal and rotate 90 CW)";
            case 8: return "Left side, bottom (Rotate 270 CW)";
            default:
                return String.valueOf(orientation);
        }
    }


從這種方法能夠清楚看到各個返回值的意思,如此我們便能夠依據實際的返回值來對圖像進行旋轉或者是鏡像處理了。


以下給出代碼用以旋轉圖片,其它的關於圖片的鏡像等處理讀者能夠依此類推:

String mPath = "D:\\XXX.JPG";
    
    File img = new File(mPath);
    BufferedImage old_img = (BufferedImage)ImageIO.read(img);  
    int w = old_img.getWidth();
    int h = old_img.getHeight();
     
    BufferedImage new_img = new BufferedImage(h,w,BufferedImage.TYPE_INT_BGR);      
    Graphics2D g2d =new_img.createGraphics();
          
    AffineTransform origXform = g2d.getTransform();
    AffineTransform newXform = (AffineTransform)(origXform.clone());
    // center of rotation is center of the panel
    
    double xRot = w/2.0;
    newXform.rotate(Math.toRadians(270.0), xRot, xRot); //旋轉270度
     
    g2d.setTransform(newXform); 
    // draw image centered in panel
    g2d.drawImage(old_img, 0, 0, null);
    // Reset to Original
    g2d.setTransform(origXform);
    
    //寫到新的文件
    FileOutputStream out = new FileOutputStream("D:\\XXX2.jpg");
    try{
        ImageIO.write(new_img, "JPG", out);
    }finally{
        out.close();
    }


注:利用上面的代碼旋轉照片后。原有照片包括的EXIF信息就不存在了。

關於該問題須要在照片旋轉之前先把EXIF信息讀出。然后再在旋轉后寫入新的照片中,能夠使用 MediaUtil 包來寫EXIF信息到圖片文件里,關於這個包的使用可參考最后的鏈接。

照片的鏡面翻轉能夠直接利用Graphic2D 的 drawImage 方法來實現:


public abstract boolean drawImage(Image img,
            int dx1,int dy1,
            int dx2,int dy2,
            int sx1,int sy1,
            int sx2,int sy2,
            ImageObserver observer);


三、補充說明


解釋部分參數的實際含義:

     Make 生產者 指產品生產廠家

    Model 型號 指設備型號

    Orientation  方向 有的相機支持,有的不支持

    X Resolution/Y Resolution X/Y方向分辨率 本欄目已有專門條目解釋此問題

    ResolutionUnit  分辨率單位 一般為PPI

    Software  軟件 顯示固件Firmware版本號

    DateTime  日期和時間

    YCbCrPositioning  色相定位

    ExifOffsetExif  信息位置。定義Exif在信息在文件里的寫入。有些軟件不顯示。

    ExposureTime 曝光時間 即快門速度

    FNumber  光圈系數

    ISO speed ratings  感光度

    ExifVersionExif  版本號

    DateTimeOriginal  創建時間

    DateTimeDigitized  數字化時間

    ComponentsConfiguration  圖像構造(多指色彩組合方案)

    CompressedBitsPerPixel(BPP)  壓縮時每像素色彩位 指壓縮程度

    ExposureBiasValue  曝光補償。

    MaxApertureValue  最大光圈

    MeteringMode  測光方式, 平均式測光、中央重點測光、點測光等。

    Lightsource  光源 指白平衡設置

    Flash  是否使用閃光燈。

    FocalLength  焦距。一般顯示鏡頭物理焦距。有些軟件能夠定義一個系數。顯示相當於35mm相機的焦距 MakerNote(User Comment)作者標記、說明、記錄

    FlashPixVersionFlashPix  版本號 (個別機型支持)

    ColorSpace 色域、色彩空間

    ExifImageWidth(Pixel X Dimension)  圖像寬度 指橫向像素數

    ExifImageLength(Pixel Y Dimension)  圖像高度 指縱向像素數


四、參考資料及資源下載

項目的最新版本號及其源代碼下載:

GitHub主頁:https://github.com/drewnoakes/metadata-extractor

http://www.drewnoakes.com/code/exif/ 

MediaUtil   http://mediachest.sourceforge.net/mediautil/


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM