什麼是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%20btn_saveExif.setOnClickListener(new%20View.OnClickListener()%20{
2
3%20@Override
4%20public%20void%20onClick(View%20v)%20{
5%20try%20{
6%20//%20tag
7%20String%20strAttr%20=%20et_attr.getText().toString().trim();
8%20//%20tag-value
9%20String%20strValue%20=%20et_value.getText().toString().trim();
10
11%20if%20(TextUtils.isEmpty(strAttr)
12%20||%20TextUtils.isEmpty(strValue))%20{
13%20Toast.makeText(MainActivity.this,%20"請填寫屬性及值",
14%20Toast.LENGTH_SHORT).show();
15%20return;
16%20}
17%20//%20擷取圖片Exif
18%20ExifInterface%20exif%20=%20new%20ExifInterface("/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資訊當中。