本文是利用Python脚本读取图片信息,有几个说明如下:
1、没有实现错误处理
2、没有读取所有信息,大概只有 GPS 信息、图片分辨率、图片像素、设备商、拍摄设备等
3、简单修改后应该能实现暴力修改图片的 GPS 信息
4、但对于本身没有 GPS 信息的图片,实现则非常复杂,需要仔细计算每个描述符的偏移量
脚本运行后,读取结果如下

脚本读取的信息
这里和 Windows 属性查看器读到的内容完全一致

图片信息1

图片信息2
源码如下
# -*- coding:utf-8 -*-
import binascii
class ParseMethod(object):
@staticmethod
def parse_default(f, count, offset):
pass
@staticmethod
def parse_latitude(f, count, offset):
old_pos = f.tell()
f.seek(12 + offset)
latitude = [0,0,0]
for i in xrange(count):
byte = f.read(4)
numerator = byte.encode(\'hex\')
byte = f.read(4)
denominator = byte.encode(\'hex\')
latitude[i] = float(int(numerator, 16)) / int(denominator, 16)
print \'Latitude:\\t%.2f %.2f\\\' %.2f\\\"\' % (latitude[0], latitude[1], latitude[2])
f.seek(old_pos)
@staticmethod
def parse_longtitude(f, count, offset):
old_pos = f.tell()
f.seek(12 + offset)
longtitude = [0,0,0]
for i in xrange(count):
byte = f.read(4)
numerator = byte.encode(\'hex\')
byte = f.read(4)
denominator = byte.encode(\'hex\')
longtitude[i] = float(int(numerator, 16)) / int(denominator, 16)
print \'Longtitude:\\t%.2f %.2f\\\' %.2f\\\"\' % (longtitude[0], longtitude[1], longtitude[2])
f.seek(old_pos)
@staticmethod
def parse_make(f, count, offset):
old_pos = f.tell()
f.seek(12 + offset)
byte = f.read(count)
a = byte.encode(\'hex\')
print \'Make:\\t\\t\' + binascii.a2b_hex(a)
f.seek(old_pos)
@staticmethod
def parse_model(f, count, offset):
old_pos = f.tell()
f.seek(12 + offset)
byte = f.read(count)
a = byte.encode(\'hex\')
print \'Model:\\t\\t\' + binascii.a2b_hex(a)
f.seek(old_pos)
@staticmethod
def parse_datetime(f, count, offset):
old_pos = f.tell()
f.seek(12 + offset)
byte = f.read(count)
a = byte.encode(\'hex\')
print \'DateTime:\\t\' + binascii.a2b_hex(a)
f.seek(old_pos)
# rational data type, 05
@staticmethod
def parse_xresolution(f, count, offset):
old_pos = f.tell()
f.seek(12 + offset)
byte = f.read(4)
numerator = byte.encode(\'hex\')
byte = f.read(4)
denominator = byte.encode(\'hex\')
xre = int(numerator, 16) / int(denominator, 16)
print \'XResolution:\\t\' + str(xre) + \' dpi\'
f.seek(old_pos)
@staticmethod
def parse_yresolution(f, count, offset):
old_pos = f.tell()
f.seek(12 + offset)
byte = f.read(4)
numerator = byte.encode(\'hex\')
byte = f.read(4)
denominator = byte.encode(\'hex\')
xre = int(numerator, 16) / int(denominator, 16)
print \'YResolution:\\t\' + str(xre) + \' dpi\'
f.seek(old_pos)
@staticmethod
def parse_exif_ifd(f, count, offset):
old_pos = f.tell()
f.seek(12 + offset)
byte = f.read(2)
a = byte.encode(\'hex\')
exif_ifd_number = int(a, 16)
for i in xrange(exif_ifd_number):
byte = f.read(2)
tag_id = byte.encode(\'hex\')
#print tag_id,
byte = f.read(2)
type_n = byte.encode(\'hex\')
#print type_n,
byte = f.read(4)
count = byte.encode(\'hex\')
#print count,
byte = f.read(4)
value_offset = byte.encode(\'hex\')
#print value_offset
value_offset = int(value_offset, 16)
EXIF_IFD_DICT.get(tag_id, ParseMethod.parse_default)(f, count, value_offset)
f.seek(old_pos)
@staticmethod
def parse_x_pixel(f, count, value):
print \'X Pixels:\\t\' + str(value)
@staticmethod
def parse_y_pixel(f, count, value):
print \'y Pixels:\\t\' + str(value)
@staticmethod
def parse_gps_ifd(f, count, offset):
old_pos = f.tell()
f.seek(12 + offset)
byte = f.read(2)
a = byte.encode(\'hex\')
gps_ifd_number = int(a, 16)
for i in xrange(gps_ifd_number):
byte = f.read(2)
tag_id = byte.encode(\'hex\')
#print tag_id,
byte = f.read(2)
type_n = byte.encode(\'hex\')
#print type_n,
byte = f.read(4)
count = byte.encode(\'hex\')
#print count,
byte = f.read(4)
value_offset = byte.encode(\'hex\')
#print value_offset
count = int(count, 16)
value_offset = int(value_offset, 16)
GPS_IFD_DICT.get(tag_id, ParseMethod.parse_default)(f, count, value_offset)
f.seek(old_pos)
IFD_dict = {
\'010f\' : ParseMethod.parse_make ,
\'0110\' : ParseMethod.parse_model ,
\'0132\' : ParseMethod.parse_datetime ,
\'011a\' : ParseMethod.parse_xresolution ,
\'011b\' : ParseMethod.parse_yresolution ,
\'8769\' : ParseMethod.parse_exif_ifd ,
\'8825\' : ParseMethod.parse_gps_ifd
}
EXIF_IFD_DICT = {
\'a002\' : ParseMethod.parse_x_pixel ,
\'a003\' : ParseMethod.parse_y_pixel
}
GPS_IFD_DICT = {
\'0002\' : ParseMethod.parse_latitude ,
\'0004\' : ParseMethod.parse_longtitude
}
with open(\'image.jpg\', \'rb\') as f:
byte = f.read(2)
a = byte.encode(\'hex\')
print \'SOI Marker:\\t\' + a
byte = f.read(2)
a = byte.encode(\'hex\')
print \'APP1 Marker:\\t\' + a
byte = f.read(2)
a = byte.encode(\'hex\')
print \'APP1 Length:\\t\' + str(int(a, 16)) + \' .Dec\'
byte = f.read(4)
a = byte.encode(\'hex\')
print \'Identifier:\\t\' + binascii.a2b_hex(a)
byte = f.read(2)
a = byte.encode(\'hex\')
print \'Pad:\\t\\t\' + a
print
print \'Begin to print Header.... \'
print \'APP1 Body: \'
byte = f.read(2)
a = byte.encode(\'hex\')
print \'Byte Order:\\t\' + a
byte = f.read(2)
a = byte.encode(\'hex\')
print \'42:\\t\\t\' + a
byte = f.read(4)
a = byte.encode(\'hex\')
print \'0th IFD Offset:\\t\' + a
print \'Finish print Header\'
print \'Begin to print 0th IFD....\'
print
#print \'Total: \',
byte = f.read(2)
a = byte.encode(\'hex\')
interoperability_number = int(a, 16)
#print interoperability_number
for i in xrange(interoperability_number):
byte = f.read(2)
tag_id = byte.encode(\'hex\')
#print tag_id,
byte = f.read(2)
type_n = byte.encode(\'hex\')
#print type_n,
byte = f.read(4)
count = byte.encode(\'hex\')
#print count,
byte = f.read(4)
value_offset = byte.encode(\'hex\')
#print value_offset
count = int(count, 16)
value_offset = int(value_offset, 16)
# simulate switch
IFD_dict.get(tag_id, ParseMethod.parse_default)(f, count, value_offset)
print
print \'Finish print 0th IFD....\'
总结
利用Python读取图片属性信息的实现方法到这就基本结束了,大家都学会了吗?希望这篇文章对大家的学习或者工作带来一定的帮助,