随着现在实景地图的如火如荼建设,无人机等航拍测绘手段的不断升级,我们在获取全景照片或者正射影像,全景视频等数据上更加快速、便捷。由于无人机本身不进行相关数据的处理,比如全景地图的生成、视频的信息解析等。以全景照片为例,无人机作业时一般会在拍摄时自动记录GPS信息,拍照的坐标信息。通过自动获取图片的经纬度信息,可以快速对照片进行定位。而我们在旅游时,通常都会进行拍照,通过开启自动记录位置后,随时可以帮助我们生成旅游地图。而这些基本信息的获取,就离不开对文件元数据(metadata)的读取。
因此,本文将重点介绍如何使用Java编程语言结合metadata-extractor去自动获取全景图片的Exif信息,获取照片的拍摄坐标信息。
元数据(Matedata),又称中介数据、中继数据,为描述数据的数据(data about data),主要是描述数据属性(property)的信息。用来支持如指示存储位置、历史数据、资源查找、文件记录等功能。元数据是关于据的组织、数据域及其关系的信息。
图片元数据(Metadata) 是嵌入到图片文件中的一些标签。比较像文件属性,但是种类繁多。对于数码图像,目前常见的研数据有EXIF, IPTC和XMP三种。
EXIF:通常被数码相机在拍摄照片时自动添加,比如相机型号、镜头、曝光、图片尺寸等信息
IPTF:比如图片标题、关键字、说明、作者、版权等信息。主要由人工在后期通过软件写入。
XMP:XMP实际上是一种元数据存储和管理的标准,可以将Exif,IPTC或其他的数据都按XMP统一的格式存放在图像文件中。
可交换图像文件(Exchangeable Image File,Exif)信息图像在拍摄时保留的相关参数:比如图像信息(厂商,分辨率等),相机拍摄记录(ISO,白平衡,饱和度,锐度等),缩略图(缩略图宽度,高度等),GPS(拍摄时的经度,纬度,高度)等,按照图像文件标准存储在图像头文件。一般使用支持图像读取的软件即可查看部分参数,但是图像如果修改,Exif信息可能丢失。

上图是一张带了坐标的JPG照片信息,在Windows中通过查看文件的详细信息,可以看到这张图片的Exif信息。
metadata-extractor 库是一个用于提取图片和视频的Exif信息的组件库。它主要提供的功能有:


更多的信息可以查看metadata-extractor 相关介绍
<!-- https://mvnrepository.com/artifact/com.drewnoakes/metadata-extractor -->
<dependency>
<groupId>com.drewnoakes</groupId>
<artifactId>metadata-extractor</artifactId>
<version>2.18.0</version>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.11</version>
<scope>test</scope>
</dependency>
这里使用ImageMetadataReader统一获取元数据信息,针对Jpg、Png、Gif等还有针对性的类,JpegMetadataReader、PngMetadataReader、GifMetadataReader等。
Metadata metadata = ImageMetadataReader.readMetadata(file);
for (Directory directory : metadata.getDirectories()) {
for (Tag tag : directory.getTags()) {
String tagName = tag.getTagName(); //标签名
String desc = tag.getDescription(); //标签信息
System.out.println(tagName + "===" + desc);//照片信息
}
}
通过代码,正常输出可以看到以下的信息:
Compression Type===Baseline
Data Precision===8 bits
Image Height===4096 pixels
Image Width===8192 pixels
Number of Components===3
Component 1===Y component: Quantization table 0, Sampling factors 2 horiz/2 vert
Component 2===Cb component: Quantization table 1, Sampling factors 1 horiz/1 vert
Component 3===Cr component: Quantization table 1, Sampling factors 1 horiz/1 vert
Image Width===8192 pixels
Image Height===4096 pixels
Bits Per Sample===8 8 8 bits/component/pixel
Image Description===default
Make===Hasselblad
Model===L1D-20c
Orientation===Top, left side (Horizontal / normal)
Samples Per Pixel===3 samples/pixel
X Resolution===72 dots per inch
Y Resolution===72 dots per inch
Resolution Unit===Inch
Software===10.00.12.07
Date/Time===2021:03:21 14:13:31
YCbCr Positioning===Center of pixel array
Windows XP Comment===0.9.142
Windows XP Keywords===pano
Exposure Time===1/200 sec
F-Number===f/5.0
Exposure Program===Program normal
ISO Speed Ratings===100
Exif Version===2.30
Date/Time Original===2021:03:21 14:13:31
Date/Time Digitized===2021:03:21 14:13:31
Components Configuration===YCbCr
Exposure Bias Value===0.3 EV
Max Aperture Value===f/2.8
Metering Mode===Average
White Balance===Daylight
Flash===Flash did not fire
Focal Length===10.3 mm
Makernote===[19829 values]
FlashPix Version===1.00
Color Space===sRGB
Exif Image Width===8192 pixels
Exif Image Height===4096 pixels
File Source===Digital Still Camera (DSC)
Scene Type===Directly photographed image
Exposure Mode===Auto exposure
White Balance Mode===Auto white balance
Digital Zoom Ratio===1
Scene Capture Type===Standard
Gain Control===None
Contrast===None
Saturation===None
Sharpness===None
Device Setting Description===0 0 0 0
Body Serial Number===0K8TGB40121511
Lens Specification===28mm f/2.8-11.0
Interoperability Index===Recommended Exif Interoperability Rules (ExifR98)
Interoperability Version===1.00
GPS Version ID===2.300
GPS Latitude Ref===N
GPS Latitude===28° 14' 37.6"
GPS Longitude Ref===E
GPS Longitude===112° 53' 24.86"
GPS Altitude Ref===Sea level
GPS Altitude===126 metres
Image Width===192 pixels
Image Height===90 pixels
Compression===JPEG
X Resolution===72 dots per inch
Y Resolution===72 dots per inch
Resolution Unit===Inch
Thumbnail Offset===21114 bytes
Thumbnail Length===18699 bytes
XMP Value Count===30
Number of Tables===4 Huffman tables
Detected File Type Name===JPEG
Detected File Type Long Name===Joint Photographic Experts Group
Detected MIME Type===image/jpeg
Expected File Name Extension===jpg
File Name===1.jpg
File Size===14057645 bytes
File Modified Date===星期二 三月 23 20:14:37 +08:00 2021
File jpegFile = new File(pathname);
Metadata metadata = JpegMetadataReader.readMetadata(jpegFile);
boolean type = metadata.containsDirectoryOfType(GpsDirectory.class);
System.out.println(type);
System.out.println(metadata.getDirectoryCount());
Iterable<Directory> it = metadata.getDirectories();
for(Directory d : it) {
System.out.println(d);
Collection<Tag> tags = d.getTags();
for(Tag tag :tags) {
System.out.println(tag.getTagName()+"==="+ tag.getDescription());
}
}
Image Width===849
Image Height===504
Bits Per Sample===8
Color Type===True Color
Compression Type===Deflate
Filter Method===Adaptive
Interlace Method===No Interlace
Detected File Type Name===PNG
Detected File Type Long Name===Portable Network Graphics
Detected MIME Type===image/png
Expected File Name Extension===png
File Name===111.png
File Size===61265 bytes
File Modified Date===星期日 十月 02 19:48:34 +08:00 2022
通过GpsDirectory来获取GeoLocation获取经坐标信息
System.out.println("开始读取gps信息...");
Collection<GpsDirectory> gpsDirectories = metadata.getDirectoriesOfType(GpsDirectory.class);
for(GpsDirectory gps : gpsDirectories) {
//获取图片的经纬度信息
GeoLocation geoLocation = gps.getGeoLocation();
System.out.println(geoLocation.getLongitude());
System.out.println(geoLocation.getLatitude());
System.out.println("********************************************************");
}
开始读取gps信息...
112.89023869444445
28.243777055555558
********************************************************
GPS Version ID===2.300
GPS Latitude Ref===N
GPS Latitude===28° 14' 37.6"
GPS Longitude Ref===E
GPS Longitude===112° 53' 24.86"
GPS Altitude Ref===Sea level
GPS Altitude===126 metres

将图片的点在地图上定位信息如上图所示
System.out.println("视频信息提取");
File file = new File("E:/静园历史影像.mp4");
Metadata metadata = Mp4MetadataReader.readMetadata(file);
for (Directory directory : metadata.getDirectories()) {
for (Tag tag : directory.getTags()) {
String tagName = tag.getTagName(); //标签名
String desc = tag.getDescription(); //标签信息
System.out.println(tagName + "===" + desc);//照片信息
}
}
视频信息提取
Major Brand===MP4 Base Media v1 [IS0 14496-12:2003]
Minor Version===512
Compatible Brands===[MP4 Base Media v1 [IS0 14496-12:2003], MP4 Base Media v2 [ISO 14496-12:2005], MP4 Base w/ AVC ext [ISO 14496-12:2005], MP4 v1 [ISO 14496-1:ch13]]
Creation Time===Fri Jan 01 08:00:00 CST 1904
Modification Time===Fri Jan 01 08:00:00 CST 1904
Duration===52608
Media Time Scale===1000
Duration in Seconds===00:00:53
Transformation Matrix===65536 0 0 0 65536 0 0 0 1073741824
Preferred Rate===1
Preferred Volume===1
Next Track ID===3
Rotation===0
Creation Time===星期五 一月 01 08:00:00 +08:00 1904
Modification Time===星期五 一月 01 08:00:00 +08:00 1904
ISO 639-2 Language Code===und
Opcolor===0 0 0
Graphics Mode===Copy
Compression Type===H.264
Width===1366 pixels
Height===768 pixels
Depth===24-bit color
Horizontal Resolution===72
Vertical Resolution===72
Frame Rate===9.905
Creation Time===星期五 一月 01 08:00:00 +08:00 1904
Modification Time===星期五 一月 01 08:00:00 +08:00 1904
ISO 639-2 Language Code===und
Balance===0
Format===MPEG-4, Advanced Audio Coding (AAC)
Number of Channels===2
Sample Size===16
Sample Rate===48000
File Name===静园历史影像.mp4
File Size===16800279 bytes
File Modified Date===星期六 七月 16 23:16:24 +08:00 2022


以上是参考了网友的博文,原文地址:Java获取图像Exif信息
官网对相关参数的定义见:metadata-extractor
以上就是今天要讲的内容,本文简单介绍了Metadata 元数据以及Exif 可交换图像文件信息的基本知识,介绍了 metadata-extractor的具体用法,展示了png图像元数据读取、GPS坐标识别和定位以及视频的元数据信息提取,而metadata-extractor提供了大量能使我们快速便捷地获取元数据的方法。
我有一个Ruby程序,它使用rubyzip压缩XML文件的目录树。gem。我的问题是文件开始变得很重,我想提高压缩级别,因为压缩时间不是问题。我在rubyzipdocumentation中找不到一种为创建的ZIP文件指定压缩级别的方法。有人知道如何更改此设置吗?是否有另一个允许指定压缩级别的Ruby库? 最佳答案 这是我通过查看rubyzip内部创建的代码。level=Zlib::BEST_COMPRESSIONZip::ZipOutputStream.open(zip_file)do|zip|Dir.glob("**/*")d
我试图在一个项目中使用rake,如果我把所有东西都放到Rakefile中,它会很大并且很难读取/找到东西,所以我试着将每个命名空间放在lib/rake中它自己的文件中,我添加了这个到我的rake文件的顶部:Dir['#{File.dirname(__FILE__)}/lib/rake/*.rake'].map{|f|requiref}它加载文件没问题,但没有任务。我现在只有一个.rake文件作为测试,名为“servers.rake”,它看起来像这样:namespace:serverdotask:testdoputs"test"endend所以当我运行rakeserver:testid时
我的目标是转换表单输入,例如“100兆字节”或“1GB”,并将其转换为我可以存储在数据库中的文件大小(以千字节为单位)。目前,我有这个:defquota_convert@regex=/([0-9]+)(.*)s/@sizes=%w{kilobytemegabytegigabyte}m=self.quota.match(@regex)if@sizes.include?m[2]eval("self.quota=#{m[1]}.#{m[2]}")endend这有效,但前提是输入是倍数(“gigabytes”,而不是“gigabyte”)并且由于使用了eval看起来疯狂不安全。所以,功能正常,
Rails2.3可以选择随时使用RouteSet#add_configuration_file添加更多路由。是否可以在Rails3项目中做同样的事情? 最佳答案 在config/application.rb中:config.paths.config.routes在Rails3.2(也可能是Rails3.1)中,使用:config.paths["config/routes"] 关于ruby-on-rails-Rails3中的多个路由文件,我们在StackOverflow上找到一个类似的问题
对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl
大约一年前,我决定确保每个包含非唯一文本的Flash通知都将从模块中的方法中获取文本。我这样做的最初原因是为了避免一遍又一遍地输入相同的字符串。如果我想更改措辞,我可以在一个地方轻松完成,而且一遍又一遍地重复同一件事而出现拼写错误的可能性也会降低。我最终得到的是这样的:moduleMessagesdefformat_error_messages(errors)errors.map{|attribute,message|"Error:#{attribute.to_s.titleize}#{message}."}enddeferror_message_could_not_find(obje
我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta
好的,所以我的目标是轻松地将一些数据保存到磁盘以备后用。您如何简单地写入然后读取一个对象?所以如果我有一个简单的类classCattr_accessor:a,:bdefinitialize(a,b)@a,@b=a,bendend所以如果我从中非常快地制作一个objobj=C.new("foo","bar")#justgaveitsomerandomvalues然后我可以把它变成一个kindaidstring=obj.to_s#whichreturns""我终于可以将此字符串打印到文件或其他内容中。我的问题是,我该如何再次将这个id变回一个对象?我知道我可以自己挑选信息并制作一个接受该信