Android--操作圖片Exif信息


前言

  在Android系統中,圖片文件在內存中以像素點的二維數組加載,存放像素信息,還會在開頭加上一些額外的照片拍攝參數信息,這些信息就是Exif。Android2.0之后,媒體庫加入了操作圖片Exif的類,本篇博客主要講解如何在Android應用中操作圖片的Exif信息。

  本篇博客主要內容:

  1. 什么是Exif
  2. ExifInterface
  3. 操作Exif

 

什么是Exif

   先來了解什么是Exif。Exif是一種圖像文件格式,它的數據存儲於JPEG格式是完全相同的,實際上Exif格式就是JPEG格式頭插入了數碼照片的信息,包括拍攝的光圈、快門、平衡白、ISO、焦距、日期時間等各種和拍攝條件以及相機品牌、型號、色彩編碼以及GPS等。簡單來說,Exif=拍攝參數+JPED。因此,可以利用任何可以查看JPEG文件的看圖軟件瀏覽Exif信息,但是並不是所有圖形程序都能處理Exif信息,而自Android2.0之后,加入了對圖片Exif數據的支持。

  

ExifInterface

   在Android下,通過ExifInterface類操作圖片的Exif信息,雖然這個類的名字包含Interface,但它不是一個接口,它是一個類,處於"android.media.ExifInterface"包下,是媒體庫的一部分功能的實現。ExifInterface有一個構造函數,接受一個String類型的數據,此為讀取圖片文件的地址。

  Exif數據在圖片中可以理解為Key-value鍵值對的方式存儲,一般通過如下幾個方法操作:

  • String getAttribute(String tag):獲取圖片中屬性為tag的字符串值。
  • double getAttribute(String tag,double defaultValue):獲取圖片中屬性為tag的double值。
  • int getAttributeInt(String tag,defaultValue):獲取圖片中屬性為tag的int值。
  • void setAttribute(String tag,String value):根據輸入參數,設定圖片Exif的值。
  • void saveAttrubutes():把內存中圖片的Exif寫入到圖片中。

  可以看到,上面大部分方法操作了一個String類型的tag參數,此為Exif的屬性,在ExifInterface中定義了一些字符串的靜態常量表示這些tag值,常用如下:

  • TAG_APERTURE:光圈值。
  • TAG_DATETIME:拍攝時間,取決於設備設置的時間。
  • TAG_EXPOSURE_TIME:曝光時間。
  • TAG_FLASH:閃光燈。
  • TAG_FOCAL_LENGTH:焦距。
  • TAG_IMAGE_LENGTH:圖片高度。
  • TAG_IMAGE_WIDTH:圖片寬度。
  • TAG_ISO:ISO。
  • TAG_MAKE:設備品牌。
  • TAG_MODEL:設備型號,整形表示,在ExifInterface中有常量對應表示。
  • TAG_ORIENTATION:旋轉角度,整形表示,在ExifInterface中有常量對應表示。

  以上常量不包括GPS的信息,實際上Exif還可以保存拍攝時GPS的信息,但是需要設備支持。下面通過一個Demo,講解一下這些參數的獲取與值的展示:

  代碼如下:

 1         btn_readExifInLog.setOnClickListener(new View.OnClickListener() {
 2 
 3             @Override
 4             public void onClick(View v) {
 5                 try {
 6                     ExifInterface exifInterface = new ExifInterface(
 7                             "/sdcard/a.jpg");
 8                     String FFNumber = exifInterface
 9                             .getAttribute(ExifInterface.TAG_APERTURE);
10                     String FDateTime = exifInterface
11                             .getAttribute(ExifInterface.TAG_DATETIME);
12                     String FExposureTime = exifInterface
13                             .getAttribute(ExifInterface.TAG_EXPOSURE_TIME);
14                     String FFlash = exifInterface
15                             .getAttribute(ExifInterface.TAG_FLASH);
16                     String FFocalLength = exifInterface
17                             .getAttribute(ExifInterface.TAG_FOCAL_LENGTH);
18                     String FImageLength = exifInterface
19                             .getAttribute(ExifInterface.TAG_IMAGE_LENGTH);
20                     String FImageWidth = exifInterface
21                             .getAttribute(ExifInterface.TAG_IMAGE_WIDTH);
22                     String FISOSpeedRatings = exifInterface
23                             .getAttribute(ExifInterface.TAG_ISO);
24                     String FMake = exifInterface
25                             .getAttribute(ExifInterface.TAG_MAKE);
26                     String FModel = exifInterface
27                             .getAttribute(ExifInterface.TAG_MODEL);
28                     String FOrientation = exifInterface
29                             .getAttribute(ExifInterface.TAG_ORIENTATION);
30                     String FWhiteBalance = exifInterface
31                             .getAttribute(ExifInterface.TAG_WHITE_BALANCE);
32 
33                     Log.i(TAG, "FFNumber:" + FFNumber);
34                     Log.i(TAG, "FDateTime:" + FDateTime);
35                     Log.i(TAG, "FExposureTime:" + FExposureTime);
36                     Log.i(TAG, "FFlash:" + FFlash);
37                     Log.i(TAG, "FFocalLength:" + FFocalLength);
38                     Log.i(TAG, "FImageLength:" + FImageLength);
39                     Log.i(TAG, "FImageWidth:" + FImageWidth);
40                     Log.i(TAG, "FISOSpeedRatings:" + FISOSpeedRatings);
41                     Log.i(TAG, "FMake:" + FMake);
42                     Log.i(TAG, "FModel:" + FModel);
43                     Log.i(TAG, "FOrientation:" + FOrientation);
44                     Log.i(TAG, "FWhiteBalance:" + FWhiteBalance);
45                 } catch (Exception e) {
46                     // TODO Auto-generated catch block
47                     e.printStackTrace();
48                 }
49             }
50         });

  獲得數據:

  

操作Exif

  上面提到,獲取與設置圖片的Exif信息,使用到的ExifInterface中的方法,上面已經列舉出來了,主要是通過tag指定存儲。

  這里說明一下,Exif信息在圖片中以二進制的形式存儲,每個字段存儲的數據位數是固定的,並且tag的數量也是固定,所以我們只能操作圖片Exif信息中已經存在的tag的值,並且保存的數據要依照它存儲位數的限制,如果存儲的數據類型錯誤,將會導致存儲的數據可能無法正確的取出,超出位數將被截取。如無法將TAG_ORIENTATION中存儲一個字符串的數據,它必須存儲int類型的值,多出來的將被截取。

  還有一點需要注意的,saveAttributes()方法主要用於把內存中所有當前Exif信息保存到目標圖片中,依照官方文檔的解釋,它是一個低效率的,它會把圖片的所有Exif信息,重新依次保存到目標圖片,所以推薦使用setAttribute()方法進行設置Exif信息。但是在實際應用中發現,如果僅使用setAttribute()設置Exif信息,將不會寫入到目標圖片中,只有在改變Exif信息后,調用saveAttribute()才可以把新的Exif寫入到目標圖片中。這個過程效率比較低,模擬器上會卡頓一下,但是真機測試沒有這樣的情況,反應很快。

  下面通過一個簡單的Demo來演示Exif的保存於讀取:

 1         btn_saveExif.setOnClickListener(new View.OnClickListener() {
 2 
 3             @Override
 4             public void onClick(View v) {
 5                 try {
 6                     // tag
 7                     String strAttr = et_attr.getText().toString().trim();
 8                     // tag-value
 9                     String strValue = et_value.getText().toString().trim();
10 
11                     if (TextUtils.isEmpty(strAttr)
12                             || TextUtils.isEmpty(strValue)) {
13                         Toast.makeText(MainActivity.this, "請填寫屬性及值",
14                                 Toast.LENGTH_SHORT).show();
15                         return;
16                     }
17                     // 獲取圖片Exif
18                     ExifInterface exif = new ExifInterface("/sdcard/a.jpg");
19                     // 保存指定tag的值
20                     exif.setAttribute(strAttr,strValue);
21                     // 把Exif信息寫入目標圖片
22                     exif.saveAttributes();
23                     Toast.makeText(MainActivity.this, "Exif信息保存成功",
24                             Toast.LENGTH_SHORT).show();
25                 } catch (Exception e) {
26                     e.printStackTrace();
27                 }
28             }
29         });
30         btn_readExif.setOnClickListener(new View.OnClickListener() {
31 
32             @Override
33             public void onClick(View v) {
34                 try {
35                     // tag
36                     String strAttr = et_attr.getText().toString().trim();
37 
38                     if (TextUtils.isEmpty(strAttr)) {
39                         Toast.makeText(MainActivity.this, "請填寫屬性",
40                                 Toast.LENGTH_SHORT).show();
41                         return;
42                     }
43                     
44                     // 獲取圖片Exif
45                     ExifInterface exif = new ExifInterface("/sdcard/a.jpg");
46                     // 獲取指定tag的屬性值
47                     String strValue = exif.getAttribute(strAttr);
48                     if (!TextUtils.isEmpty(strValue)) {
49                         Toast.makeText(MainActivity.this, strAttr+"="+strValue,
50                                 Toast.LENGTH_SHORT).show();
51                     } else {
52                         Toast.makeText(MainActivity.this, "圖片Exif中沒有屬性值為"+strAttr+"的信息",
53                                 Toast.LENGTH_SHORT).show();
54                     }
55                 } catch (Exception e) {
56                     e.printStackTrace();
57                 }
58             }
59         });

  效果展示,先讀取Make信息,再寫入Make信息並重新讀取:

  注意,上面示例中,如果Attribute寫任意值,會提示保存成功,但是並沒有寫入到目標圖片的Exif信息當中。

 

  源碼下載

 

總結

  以上就是Exif的所有信息,其實很好理解,就是圖片中蘊含的一些拍攝環境的信息。如果開發一款與攝影相關的軟件,Exif的信息應該是會用的到的。

 


免責聲明!

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



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