android 獲取圖片資訊 之 ExifInterface
Android--操作圖片Exif資訊
---------------------------------------------------------------------------------------
作者:承香墨影
出處:http://plokmju.cnblogs.com/
更多內容,請閱讀本人新書:《Android深入淺出》
歡迎轉載,但還請尊重勞動果實,保留此段宣告並註明原文連結。
---------------------------------------------------------------------------------------
前言
在Android系統中,圖片檔案在記憶體中以畫素點的二維陣列載入,存放畫素資訊,還會在開頭加上一些額外的照片拍攝引數資訊,這些資訊就是Exif。Android2.0之後,媒體庫加入了操作圖片Exif的類,本篇部落格主要講解如何在Android應用中操作圖片的Exif資訊。
本篇部落格主要內容:
- 什麼是Exif
- ExifInterface
- 操作Exif
先來了解什麼是Exif。Exif是一種影象檔案格式,它的資料儲存於JPEG格式是完全相同的,實際上Exif格式就是JPEG格式頭插入了數碼照片的資訊,包括拍攝的光圈、快門、平衡白、ISO、焦距、日期時間等各種和拍攝條件以及相機品牌、型號、色彩編碼以及GPS等。簡單來說,Exif=拍攝引數+JPED。因此,可以利用任何可以檢視JPEG檔案的看圖軟體瀏覽Exif資訊,但是並不是所有圖形程式都能處理Exif資訊,而自Android2.0之後,加入了對圖片Exif資料的支援。
Exif,是英文Exchangeable Image file(可交換影象檔案)的縮寫,Exif檔案實際上可以看作是JPEG影象檔案格式的一種,並且遵從JPEG檔案格式標準。Exif資訊就是由數碼相機在拍攝過程中採集一系列相互聯絡的拍攝資訊,然後把這些資訊放置在我們所熟知的JPEG格式檔案原始資料的內部,也就是說Exif資訊是鑲嵌在JPEG影象檔案格式內的一組拍攝引數,而這些引數主要包括拍攝時的光圈、快門、ISO值、拍攝日期間等各種與當時攝影條件相關的資訊,相機品牌型號,色彩編碼,拍攝時錄製的聲音檔案甚至全球定位系統(GPS)等資訊。
簡單的說,它就好像是傳統相機日期後背具有的日期列印功能一樣,只不過Exif所記錄的資訊引數更為詳細和全面。也因此,理論上只要支援JPEG檔案格式的影象處理軟體都可以用來觀看或者修改Exif檔案資訊,不過,如果修改了圖片,原始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寫入到圖片中。
- 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中有常量對應表示。
程式碼如下:
- btn_readExifInLog.setOnClickListener(new View.OnClickListener() {
- @Override
- publicvoid onClick(View v) {
- try {
- ExifInterface exifInterface = new ExifInterface(
- "/sdcard/a.jpg");
- String FFNumber = exifInterface
- .getAttribute(ExifInterface.TAG_APERTURE);
- String FDateTime = exifInterface
- .getAttribute(ExifInterface.TAG_DATETIME);
- String FExposureTime = exifInterface
- .getAttribute(ExifInterface.TAG_EXPOSURE_TIME);
- String FFlash = exifInterface
- .getAttribute(ExifInterface.TAG_FLASH);
- String FFocalLength = exifInterface
- .getAttribute(ExifInterface.TAG_FOCAL_LENGTH);
- String FImageLength = exifInterface
- .getAttribute(ExifInterface.TAG_IMAGE_LENGTH);
- String FImageWidth = exifInterface
- .getAttribute(ExifInterface.TAG_IMAGE_WIDTH);
- String FISOSpeedRatings = exifInterface
- .getAttribute(ExifInterface.TAG_ISO);
- String FMake = exifInterface
- .getAttribute(ExifInterface.TAG_MAKE);
- String FModel = exifInterface
- .getAttribute(ExifInterface.TAG_MODEL);
- String FOrientation = exifInterface
- .getAttribute(ExifInterface.TAG_ORIENTATION);
- String FWhiteBalance = exifInterface
- .getAttribute(ExifInterface.TAG_WHITE_BALANCE);
- Log.i(TAG, "FFNumber:" + FFNumber);
- Log.i(TAG, "FDateTime:" + FDateTime);
- Log.i(TAG, "FExposureTime:" + FExposureTime);
- Log.i(TAG, "FFlash:" + FFlash);
- Log.i(TAG, "FFocalLength:" + FFocalLength);
- Log.i(TAG, "FImageLength:" + FImageLength);
- Log.i(TAG, "FImageWidth:" + FImageWidth);
- Log.i(TAG, "FISOSpeedRatings:" + FISOSpeedRatings);
- Log.i(TAG, "FMake:" + FMake);
- Log.i(TAG, "FModel:" + FModel);
- Log.i(TAG, "FOrientation:" + FOrientation);
- Log.i(TAG, "FWhiteBalance:" + FWhiteBalance);
- } catch (Exception e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- });
操作Exif
上面提到,獲取與設定圖片的Exif資訊,使用到的ExifInterface中的方法,上面已經列舉出來了,主要是通過tag指定儲存。
這裡說明一下,Exif資訊在圖片中以二進位制的形式儲存,每個欄位儲存的資料位數是固定的,並且tag的數量也是固定,所以我們只能操作圖片Exif資訊中已經存在的tag的值,並且儲存的資料要依照它儲存位數的限制,如果儲存的資料型別錯誤,將會導致儲存的資料可能無法正確的取出,超出位數將被擷取。如無法將TAG_ORIENTATION中儲存一個字串的資料,它必須儲存int型別的值,多出來的將被擷取。
還有一點需要注意的,saveAttributes()方法主要用於把記憶體中所有當前Exif資訊儲存到目標圖片中,依照官方文件的解釋,它是一個低效率的,它會把圖片的所有Exif資訊,重新依次儲存到目標圖片,所以推薦使用setAttribute()方法進行設定Exif資訊。但是在實際應用中發現,如果僅使用setAttribute()設定Exif資訊,將不會寫入到目標圖片中,只有在改變Exif資訊後,呼叫saveAttribute()才可以把新的Exif寫入到目標圖片中。這個過程效率比較低,模擬器上會卡頓一下,但是真機測試沒有這樣的情況,反應很快。
下面通過一個簡單的Demo來演示Exif的保存於讀取:
- btn_saveExif.setOnClickListener(new View.OnClickListener() {
- @Override
- publicvoid onClick(View v) {
- try {
- // tag
- String strAttr = et_attr.getText().toString().trim();
- // tag-value
- String strValue = et_value.getText().toString().trim();
- if (TextUtils.isEmpty(strAttr)
- || TextUtils.isEmpty(strValue)) {
- Toast.makeText(MainActivity.this, "請填寫屬性及值",
- Toast.LENGTH_SHORT).show();
- return;
- }
- // 獲取圖片Exif
- ExifInterface exif = new ExifInterface("/sdcard/a.jpg");
- // 儲存指定tag的值
- exif.setAttribute(strAttr,strValue);
- // 把Exif資訊寫入目標圖片
- exif.saveAttributes();
- Toast.makeText(MainActivity.this, "Exif資訊儲存成功",
- Toast.LENGTH_SHORT).show();
- } catch (Exception e) {
- e.printStackTrace();
- }
- }
- });
- btn_readExif.setOnClickListener(new View.OnClickListener() {
- @Override
- publicvoid onClick(View v) {
- try {
- // tag
- String strAttr = et_attr.getText().toString().trim();
- if (TextUtils.isEmpty(strAttr)) {
- Toast.makeText(MainActivity.this, "請填寫屬性",
- Toast.LENGTH_SHORT).show();
- return;
- }
- // 獲取圖片Exif
- ExifInterface exif = new ExifInterface("/sdcard/a.jpg");
- // 獲取指定tag的屬性值
- String strValue = exif.getAttribute(strAttr);
- if (!TextUtils.isEmpty(strValue)) {
- Toast.makeText(MainActivity.this, strAttr+"="+strValue,
- Toast.LENGTH_SHORT).show();
- } else {
- Toast.makeText(MainActivity.this, "圖片Exif中沒有屬性值為"+strAttr+"的資訊",
- Toast.LENGTH_SHORT).show();
- }
- } catch (Exception e) {
- e.printStackTrace();
- }
- }
- });