android 获取图片信息 之 ExifInterface

article/2025/9/25 23:44:16

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


http://chatgpt.dhexx.cn/article/GNtghFCn.shtml

相关文章

图片头文件信息解析

今天发现获取图片的宽高,并不需要把图片完全读完之后再来获取,而只需要读取文件头文件,几十个字节便可以读出文件的宽高。 图片的文件头部存储有该图片相关信息,可以从中读取相应字段,得到尺寸、大小、格式等信息。由于…

图片Exif信息解析(Java实现)

前言 可交换图像文件(Exchangeable Image File,Exif)信息图像在拍摄时保留的相关参数:比如图像信息(厂商,分辨率等),相机拍摄记录(ISO,白平衡,饱和度,锐度等…

java项目实战:处理图片水印,提取图片信息,生成excel表

在这次应用软件设计课程中,要求从今年的软件杯大赛上的项目选择一个实现。我选的是"网店工商信息提取",具体要求就是:从给出的带水印的图片中提取出企业名称和企业注册号,并根据这些信息生成excel表格。 刚刚开始以为这…

nodejs图片读取

response返回都是html/text,向前台输出一张图片用的image/jpeg,服务器读取图片的时候是按照binary的二进制方式读取,给客户端返回的时候也按照binary二进制的方式返回。 从服务器读取一张图片给客户端输出: 效果:输入localhost:…

[软件工具] 如何批量获取图片信息,尺寸、大小、路径、文件名,然后导出表格或者txt的文本,下面教你使用方法

前几天遇到一个比较棘手的需求: 如何获取几万张图片的大量的图片信息,如尺寸、大小、路径、文件名等等, 去看了百度 好多都是教写批处理的文件信息,对批处理不是很懂,写了几次都没成功 然后做这么一款软件&#xff…

APICloud框架——获取本地图片信息

api.getPicture 获取本地图片放置到服务器上或者在app中预览是app的基本功能,今天使用了APICloud框架的api.getPicture这个api获取到的本地图片预览在app中,就像上传qq头像一样,其实就是这个需求,获取本地照片(拍摄照片…

imagemagick 获取图片信息,放大缩小,指定区域,旋转,边框,draw png压缩问题

Table of Contents 1.获取图片信息 2.放大缩小 -resize 3.放大缩小 -sample 4.指定区域 5.旋转 6.添加边框 7.draw的用法 1.获取图片信息 C:\Users\Administrator>magick identify F:\imagemagick\1.jpg F:\imagemagick\1.jpg JPEG 1440x2560 1440x256000 8-bit sRGB …

解决透视变换后图片信息丢失的问题

问题背景:最近在做图像拼接,思路是首先对输入的两幅图进行特征提取,提取的方法使用的是经典的SIFT(Scale-invariant feature transform)算法;然后进行特征匹配,匹配的思路是将一幅图中的特征点以…

wx.getImageInfo(Object object)获取图片信息

微信小程序第一周总结 总结人:陈曦 wx.getImageInfo(Object object)获取图片信息 在小程序/小游戏中使用网络相关的 API 时,每个微信小程序需要事先设置一个通讯域名,小程序只可以跟指定的域名与进行网络通信。包括普通 HTTPS 请求&#xff…

利用Python实现图片信息隐藏

最近上了一门信息隐藏的课,讲的关于技术层面的内容还是蛮多的,但是我也没记住几个。吭吭,言归正传,这两天要交大作业,自己手动实现一种图像信息隐藏,查了一番资料后,决定Python来做(主要是调用包方便),原理很简单,主要利用修改图像RGB通道的某一通道值来隐藏二维信息…

Python获取图片信息小栗子

1.首先我们需要先导入所需要的包,没有的话可以【 pip install ~】 来获取 import requests from lxml import etree 2.接下来我们要进行UA伪装,伪装的目的就是把电脑伪装成人 因为很多wangzahn都有反扒机制,不进行伪装的话根本就无法进行爬…

线性回归用matlab怎么做,matlab中如何应用regress()函数进行线性回归分析?

matlab中如何应用regress()函数进行线性回归分析?回归分析是研究一个随机变量与一个或多个普通变量之间的相关系的统计方法。如果做回归分析,有很多软件都已经封装好了的,我们只需直接调用就可以了。例如Matlab中的regress()就是做线性回归分…

[Matlab]篇----回归分析Matlab命令(regress篇)

[Matlab]篇—-回归分析Matlab命令(regress篇) 一、简介 最近在做回归分析方面的东西,网上查阅相关资料,通过实际调试,对调试结果进行总结。 回归分析法指利用数据统计原理,对大量统计数据进行数学处理,并…

回归(Regression )

回归(Regression ) 模型定义 Regression 就是找到一个函数 functionfunction ,通过输入特征 x,输出一个数值 Scalar。 模型应用 股市预测(Stock market forecast) 输入:过去10年股票的变动、…

数学建模:回归分析——regress 函数

matlab中regress 函数 https://jingyan.baidu.com/article/ca41422f6777aa1eae99ed87.htmlhttps://jingyan.baidu.com/article/ca41422f6777aa1eae99ed87.html 结果 附录 1、数据文本:data1_1 0 68 110 121 23 111 100 37 66 110 12 169 50 46 87 117 16 155 25…

[matlab]多元线性回归-regress函数的用法

转载博客原址:https://blog.csdn.net/YinJianxiang/article/details/81464245 回归分析 回归分析 (英语:Regression Analysis) 是一种统计学上分析数据的方法,目的在于了解两个或多个变量间是否相关、相关方向与强度,并建立数学…

机器学习——regression

1 用线性回归找到最佳拟合直线 线性回归 优点:结果易于理解,计算上不复杂 缺点:对非线性的数据拟合不好 使用数据类型:数值型数据和标称型数据。 回归方程:是根据样本资料通过回归分析所得到的反映一个变量&#xff0…

回归问题(Regression)

Regression 前言Dependent vs. Explanatory VariablesHandle Numerical Labelssquared error和variance什么区别 Linear RegressionLinear Regression in 1 Dimension Least Squares (最小二乘,重点)Least Squares ObjectiveMinimizing a Dif…

Fama-Macbeth回归图形化:EAP.fama_macbeth.Fama_macbeth_regress.plot()

实证资产定价(Empirical asset pricing)已经发布于Github和Pypi. 包的具体用法(Documentation)博主将会陆续在CSDN中详细介绍,也可以通过Pypi直接查看。 Pypi: pip install --upgrade EAP HomePage: EAP Empirical Asset Prici…

[MATLAB]一元线性回归(regress参数检验说明)

今天学习了统计回归模型,统计回归模型是基于统计理论建立的最基本最常用的一类数据驱动模型。本篇博文主要围绕regress函数的参数进行讲解! 一元线性回归模型的概念 这也要从一个女装公司讲起! 问:请用函数关系描述身高与腿长的…