前言

  在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,讲解一下这些参数的获取与值的展示:

  代码如下:

         btn_readExifInLog.setOnClickListener(new View.OnClickListener() {

             @Override
public 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 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
public void 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
public void 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();
}
}
});

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

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

  源码下载

总结

  以上就是Exif的所有信息,其实很好理解,就是图片中蕴含的一些拍摄环境的信息。如果开发一款与摄影相关的软件,Exif的信息应该是会用的到的。

Android--操作图片Exif信息的更多相关文章

  1. Android 获取图片exif信息

    使用android api读取图片的exif信息 布局代码: <LinearLayout xmlns:android="http://schemas.android.com/apk/r ...

  2. Android 图片Exif信息相关的获取与修改

    1 Exif是什么 Exif是一种图像文件格式,它的数据存储于JPEG格式是完全相同的,实际上Exif格式就是JPEG格式头插入了 数码照片的信息,包括拍摄的光圈.快门.平衡白.ISO.焦距.日期时间 ...

  3. Android -- 加载大图片到内存,从gallery获取图片,获取图片exif信息

    1. 加载大图片到内存,从gallery获取图片 android默认的最大堆栈只有16M, 图片像素太高会导致内存不足的异常, 需要将图片等比例缩小到适合手机屏幕分辨率, 再加载. 从gallery ...

  4. 图片Exif信息

    Exif文件格式简述链接:https://www.zhihu.com/question/23727439/answer/25467748 可交换图像文件常被简称为Exif(Exchangeable i ...

  5. 七牛--关于图片上传方向不统一的问题--主要关于图片EXIF信息中旋转参数Orientation的理解

    [图片引用方向纠正]直接在图片后面添加 ?imageMogr/auto-orient eg:http://data.upfitapp.com/data/2016/10/18/1629114767606 ...

  6. 图片Exif 信息中Orientation的理解和对此的处理

    这个问题是在用七牛上传图片后获取宽高时发现的,一张图片,用图片浏览器打开始终是竖图,但是查看属性或者用七牛获取宽高,却发现宽大于高,也就是在属性中这是个横图.这样导致客户端用该宽高来展示图片会出现问题 ...

  7. 七牛:关于图片 EXIF 信息中旋转参数 Orientation 的理解

    EXIF(Exchangeable Image File)是 “可交换图像文件” 的缩写,当中包含了专门为数码相机的照片而定制的元数据,可以记录数码照片的拍摄参数.缩略图及其他属性信息,简单来说,Ex ...

  8. 改动图片exif信息

    我们先了解一下EXIF: EXIF能够附加于JPEG.TIFF.RIFF等文件之中.为其添加有关数码相机拍摄信息的内容和索引图或图像处理软件的版本号信息. 全部的JPEG文件以字符串"0xF ...

  9. Java读取图片exif信息实现图片方向自动纠正

    起因 一个对试卷进行OCR识别需求,需要实现一个功能,一个章节下的题目图片需要上下拼接合成一张大图,起初写了一个工具实现图片的合并,程序一直很稳定的运行着,有一反馈合成的图片方向不对,起初怀疑是本身图 ...

随机推荐

  1. Extjs在树上加右键菜单--2019-04-15

    效果图如下: 使用规则:将监听加到按钮或树上,监听代码如下. 代码如下: listeners : { //节点单击事件 'rowcontextmenu' : function(view, record ...

  2. ZOJ 2588 Burning Bridges 割边(处理重边)

    <题目链接> 题目大意: 给定一个无向图,让你尽可能的删边,但是删边之后,仍然需要保证图的连通性,输出那些不能被删除的边. 解题分析: 就是无向图求桥的题目,主要是提高一下处理重边的姿势. ...

  3. 福州大学软件工程1916|W班 第3次作业成绩排名

    作业链接 结对第二次-文献摘要热词统计及进阶需求 评分细则 本次作业由三部分组成(程序满分80,博客满分70,工程能力满分30) 程序评分标准 基础需求 共有7个测试用例,每个满分20分并按照一定的映 ...

  4. Metasploit中aggregator插件无法使用

    Metasploit中aggregator插件无法使用   aggregator是Metasploit自带的一个插件,用来管理会话Session.该插件使用metasploit-aggreator库. ...

  5. 用kattle将数据从SQLserver中导入到vertica中

    今天简单的学习了一下ETL工具kattle了,只是简单的上手,不过这也已经够我去做POC了. 首先大体介绍一下kattle,Kettle是一款国外开源的ETL工具,纯java编写,可以在Window. ...

  6. 4.28Linux(6)

    2019-4-28 21:27:41 明天回家.回家继续学Linux还好有个服务器!!!感觉有个属于自己的服务器感觉好爽啊!! 越努力越幸运!永远不要高估自己!!! Nginx安装 服务器的请求原理 ...

  7. vim 常用指令

    其他命令 <c-L> 重绘屏幕 <c-z> 挂起vim回到shell,想继续vim只需要输入 fg <c-x-f> 文件路径提示 <c-N> 当前文件中 ...

  8. Vue取消eslint语法限制

    话不多说,先上图: 当然,这里的警告我是知道怎么回事,原来eslint是一个语法检查工具,但是限制很严格,在我的vue文件里面很多空格都会导致红线警告(可以屏蔽),虽然可以屏蔽,但是在编译的时候老是会 ...

  9. C++之几个最常

    1.同类对象间的数据共享——静态成员 静态数据成员声明静态数据成员要采用关键字static:类静态数据成员的定义和初始化定义:static 数据类型 成员名:初始化:数据类型 类名::静态数据成员名= ...

  10. 流媒体协议(一):HLS 协议

    一.HLS 概述 HLS 全称是 HTTP Live Streaming,是一个由 Apple 公司提出的基于 HTTP 的媒体流传输协议,用于实时音视频流的传输.目前HLS协议被广泛的应用于视频点播 ...