在Android下通过ExifInterface类操作图片的Exif信息

时间:2024-01-09 10:08:02

什么是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 });

  获得数据:

在Android下通过ExifInterface类操作图片的Exif信息

  操作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信息并重新读取:

在Android下通过ExifInterface类操作图片的Exif信息在Android下通过ExifInterface类操作图片的Exif信息 在Android下通过ExifInterface类操作图片的Exif信息

  注意,上面示例中,如果Attribute写任意值,会提示保存成功,但是并没有写入到目标图片的Exif信息当中。