前言

  在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的信息应该是会用的到的。

 

版权声明:本文为plokmju原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/plokmju/p/android_exif.html