视频1 视频21 视频41 视频61 视频文章1 视频文章21 视频文章41 视频文章61 推荐1 推荐3 推荐5 推荐7 推荐9 推荐11 推荐13 推荐15 推荐17 推荐19 推荐21 推荐23 推荐25 推荐27 推荐29 推荐31 推荐33 推荐35 推荐37 推荐39 推荐41 推荐43 推荐45 推荐47 推荐49 关键词1 关键词101 关键词201 关键词301 关键词401 关键词501 关键词601 关键词701 关键词801 关键词901 关键词1001 关键词1101 关键词1201 关键词1301 关键词1401 关键词1501 关键词1601 关键词1701 关键词1801 关键词1901 视频扩展1 视频扩展6 视频扩展11 视频扩展16 文章1 文章201 文章401 文章601 文章801 文章1001 资讯1 资讯501 资讯1001 资讯1501 标签1 标签501 标签1001 关键词1 关键词501 关键词1001 关键词1501 专题2001
Python读取图片属性信息的实现方法
2020-11-27 14:16:58 责编:小采
文档
 本文是利用Python脚本读取图片信息,有几个说明如下:

1、没有实现错误处理

2、没有读取所有信息,大概只有 GPS 信息、图片分辨率、图片像素、设备商、拍摄设备等

3、简单修改后应该能实现暴力修改图片的 GPS 信息

4、但对于本身没有 GPS 信息的图片,实现则非常复杂,需要仔细计算每个描述符的偏移量

脚本运行后,读取结果如下

这里和 Windows 属性查看器读到的内容完全一致

源码如下

# -*- 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:	%.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:	%.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:	' + 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:	' + 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:	' + 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:	' + 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:	' + 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:	' + str(value)
 
 @staticmethod
 def parse_y_pixel(f, count, value):
 print 'y Pixels:	' + 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:	' + a
 
 byte = f.read(2)
 a = byte.encode('hex')
 print 'APP1 Marker:	' + a
 
 byte = f.read(2)
 a = byte.encode('hex')
 print 'APP1 Length:	' + str(int(a, 16)) + ' .Dec'
 
 byte = f.read(4)
 a = byte.encode('hex')
 print 'Identifier:	' + binascii.a2b_hex(a)
 
 byte = f.read(2)
 a = byte.encode('hex')
 print 'Pad:	' + a 
 
 print
 
 print 'Begin to print Header.... '
 print 'APP1 Body: '
 
 byte = f.read(2)
 a = byte.encode('hex')
 print 'Byte Order:	' + a 
 
 byte = f.read(2)
 a = byte.encode('hex')
 print '42:	' + a 
 
 byte = f.read(4)
 a = byte.encode('hex')
 print '0th IFD Offset:	' + 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读取图片属性信息的实现方法到这就基本结束了,大家都学会了吗?希望这篇文章对大家的学习或者工作带来一定的帮助,

更多Python读取图片属性信息的实现方法相关文章请关注PHP中文网!

下载本文
显示全文
专题