Android--操作图片Exif信息

---------------------------------------------------------------------------------------

作者:承香墨影
出处:http://plokmju.cnblogs.com/
更多内容,请阅读本人新书:《Android深入浅出》
欢迎转载,但还请尊重劳动果实,保留此段声明并注明原文链接。

---------------------------------------------------------------------------------------

前言

  在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数据的支持。

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写入到图片中。
  可以看到,上面大部分方法操作了一个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,讲解一下这些参数的获取与值的展示:
  代码如下:
btn_readExifInLog.setOnClickListener(new View.OnClickListener() {@Overridepublic void 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 blocke.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() {@Overridepublic void onClick(View v) {try {// tagString strAttr = et_attr.getText().toString().trim();// tag-valueString strValue = et_value.getText().toString().trim();if (TextUtils.isEmpty(strAttr)|| TextUtils.isEmpty(strValue)) {Toast.makeText(MainActivity.this, "请填写属性及值",Toast.LENGTH_SHORT).show();return;}// 获取图片ExifExifInterface 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() {@Overridepublic void onClick(View v) {try {// tagString strAttr = et_attr.getText().toString().trim();if (TextUtils.isEmpty(strAttr)) {Toast.makeText(MainActivity.this, "请填写属性",Toast.LENGTH_SHORT).show();return;}// 获取图片ExifExifInterface 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();}}});

效果展示,先读取Make信息,再写入Make信息并重新读取:

 

android 获取图片信息 之 ExifInterface相关推荐

  1. android android 修改 jpg exif 属性,Android开发之使用ExifInterface获取拍照后的图片属性...

    本文实例讲述了Android开发之使用ExifInterface获取拍照后的图片属性.分享给大家供大家参考,具体如下: ExifInterface exif = new ExifInterface(f ...

  2. Android获取手机本地图片并显示

    一.序言 在安卓开发过程中,有时候我们的应用需要使用手机本地图片,这就需要本地图片访问权限以及相关的获取方法,本文将手机本地图片的获取流程和代码做了一个总结,希望能够对大家有一定帮助: 二.功能分析 ...

  3. android屏幕密度高度,Android获取常用辅助方法(获取屏幕高度、宽度、密度、通知栏高度、截图)...

    我们需要获取Android手机或Pad的屏幕的物理尺寸,以便于界面的设计或是其他功能的实现.下面就分享一下Android中常用的一些辅助方法: 获取屏幕高度: /** * 获得屏幕高度 * @para ...

  4. Android获取当前时间

    2019独角兽企业重金招聘Python工程师标准>>> Android获取当前时间 2012-01-09 17:29:55|  分类: 私人日志 |字号 订阅 [转自stay] 取得 ...

  5. Android获取设备状态栏status bar高度的正确姿势

    Android获取设备状态栏高度的正确姿势 正确代码方式: int height = 0;int resourceId = getApplicationContext().getResources() ...

  6. 格式android id,android 获取APP的唯一标识applicationId的实例

    使用getIdentifier()方法可以方便的获各应用包下的指定资源ID. 方式一 int indentify = getResources().getIdentifier("com.te ...

  7. android 获取图片

    Android获取手机或者内存卡里面的图片有两种方式 1.这是通过一种action Intent intent=new Intent();intent.setAction(Intent.ACTION_ ...

  8. android 获取短信验证码倒计时

     android 获取短信验证码倒计时 public class MainActivity extends Activity { private Button submit;     privat ...

  9. [置顶] Android代码----android获取3G或wifi流量信息

    android获取3G或wifi流量信息: IBatteryStats battryStats = IBatteryStats.Stub.asInterface(ServiceManager.getS ...

  10. android 获取视频缩略图终极解决方案(ffmpeg)

    android 获取视频缩略图终极解决方案(ffmpeg) 参考文章: (1)android 获取视频缩略图终极解决方案(ffmpeg) (2)https://www.cnblogs.com/juka ...

最新文章

  1. python 第一行输入n表示一天中有多少人买水果_Python编程:从入门到实践——【作业】——第五章作业...
  2. html金额输入框转大写,纯CSS实现输入框字符自动转为小写或大写
  3. 有三AI一周年了,说说我们的初衷,生态和愿景
  4. Angular之组件的创建
  5. OpenStack tokens id获取测试
  6. stauml工具怎么导入文件_小伙教大家怎么剪辑短视频,1小时就学会添加字幕,值得收藏哦...
  7. ARCENGINE 10 开发遇到的一些问题
  8. java中是否可以覆盖over_”static”关键字是什么意思?Java中是否可以覆盖(override)一个private或者是static的方法?...
  9. asp.net mvc 2.0 TryValidateModel(Object)方法
  10. 【数据分析】基于matlab GUI学生成绩查询系统【含Matlab源码 604期】
  11. 游戏设计类毕业论文文献(推荐10篇)
  12. 5. DICOM图像层级分类-DCMTK-压缩图像PixelData读取
  13. windows下搭建tracker服务器
  14. 计算机科学导论第五版第二章答案,(计算机科学导论第2章答案.docx
  15. 40163 php,【PHP】微信支付JsApi 40163错误
  16. 移动应用/APP的测试流程及方法
  17. 2023年品牌惊蛰节气海报赏析
  18. 宽带波束形成-----恒定波束宽度设计
  19. Cortex-A55核心板的温升实测!
  20. 【计算机二级等级考试】公共基础知识-学习笔记

热门文章

  1. Linux电源管理(7)_Wakeup events framework
  2. 并发编程 - lost wakeup
  3. 转(js幻灯片,jQuery幻灯,js焦点轮换图,js幻灯轮播代码大全(2) - 酷站代码)...
  4. 龙讯7号 国芯发布龙芯电脑引争议
  5. 2022年考 PMP 证书有什么意义?
  6. 电脑重装系统后,重启时遇到错误,报错需要重新启动,并重新安装系统
  7. mysql中文显示标题列_我在数据库中建立的列名为英文的,但标题是中文的,在编程界面中显示的是英文的列名,怎样设置成中文的?...
  8. 关于安卓毛玻璃实现(一)动态毛玻璃
  9. html progress改变样式,自定义html中Progress的样式
  10. php计算工资的代码,php计算税后工资的方法_PHP