如何从 HEIC 文件中提取 GPS 位置?

LA_*_*LA_ 1 exif python-3.x heic

我使用以下代码从 JPG 和 HEIC 文件中提取 GPS 位置:

#coding=utf-8
from PIL import Image
from urllib.request import urlopen
from PIL.ExifTags import TAGS
from PIL.ExifTags import GPSTAGS

from pillow_heif import register_heif_opener

def get_exif(filename):
    image = Image.open(filename)
    image.verify()
    return image._getexif()

def get_geotagging(exif):
    if not exif:
        raise ValueError("No EXIF metadata found")

    geotagging = {}
    for (idx, tag) in TAGS.items():
        if tag == 'GPSInfo':
            if idx not in exif:
                raise ValueError("No EXIF geotagging found")

            for (key, val) in GPSTAGS.items():
                if key in exif[idx]:
                    geotagging[val] = exif[idx][key]

    return geotagging

register_heif_opener()

my_image='IMG_9610.HEIC'
#my_image='IMG_20210116_215317.jpg'

exif = get_exif(my_image)
labeled = get_geotagging(exif)
print(labeled)
Run Code Online (Sandbox Code Playgroud)

此代码适用于 JPEG 文件,但使用 HEIC 将返回以下错误:

AttributeError: _getexif
Run Code Online (Sandbox Code Playgroud)

如果我添加以下功能

def get_labeled_exif(exif):
    labeled = {}
    for (key, val) in exif.items():
        labeled[TAGS.get(key)] = val

    return labeled
Run Code Online (Sandbox Code Playgroud)

并将 '_getexif()' 替换为 'getexif()' 那么它适用于两个文件,但数据在那里加密 -'GPSInfo': 1234并且get_geotagging()不适用于此类 exif。

我该如何解决它?

Lif*_*lex 5

更新帖子 2022 年 6 月 12 日

下面的代码能够从我系统上的 HEIC 图像文件中提取 GEO 标记信息。

from PIL import Image
from pillow_heif import register_heif_opener


def get_exif(filename):
    image = Image.open(filename)
    image.verify()
    return image.getexif().get_ifd(0x8825)


def get_geotagging(exif):
    geo_tagging_info = {}
    if not exif:
        raise ValueError("No EXIF metadata found")
    else:
        gps_keys = ['GPSVersionID', 'GPSLatitudeRef', 'GPSLatitude', 'GPSLongitudeRef', 'GPSLongitude',
                    'GPSAltitudeRef', 'GPSAltitude', 'GPSTimeStamp', 'GPSSatellites', 'GPSStatus', 'GPSMeasureMode',
                    'GPSDOP', 'GPSSpeedRef', 'GPSSpeed', 'GPSTrackRef', 'GPSTrack', 'GPSImgDirectionRef',
                    'GPSImgDirection', 'GPSMapDatum', 'GPSDestLatitudeRef', 'GPSDestLatitude', 'GPSDestLongitudeRef',
                    'GPSDestLongitude', 'GPSDestBearingRef', 'GPSDestBearing', 'GPSDestDistanceRef', 'GPSDestDistance',
                    'GPSProcessingMethod', 'GPSAreaInformation', 'GPSDateStamp', 'GPSDifferential']

        for k, v in exif.items():
            try:
                geo_tagging_info[gps_keys[k]] = str(v)
            except IndexError:
                pass
        return geo_tagging_info


register_heif_opener()

my_image = 'IMG_8362.heic'
image_info = get_exif(my_image)
results = get_geotagging(image_info)
print(results)
# x used to mask data
{'GPSLatitudeRef': 'N', 
'GPSLatitude': '(3x.0, 5x.0, 1x.0x)', 
'GPSLongitudeRef': 'W', 
'GPSLongitude': '(8x.0, 2x.0, 5x.2x)', 
'GPSAltitudeRef': "b'\\x00'", 
'GPSAltitude': '279.63243243243244', 
'GPSSpeedRef': 'K', 
'GPSSpeed': '0.04649941997239198', 
'GPSImgDirectionRef': 'T', 
'GPSImgDirection': '274.37165833514456', 
'GPSDestBearingRef': 'T', 
'GPSDestBearing': '27x.37165833514456', 
'GPSDateStamp': '2022:06:12'}
Run Code Online (Sandbox Code Playgroud)
----------------------------------------
My system information
----------------------------------------
Platform:     Apple
OS Version:   macOS Catalina 10.15.7
Python: 3.9
Pillow: 9.1.1
pillow_heif: 0.3.0
----------------------------------------
Run Code Online (Sandbox Code Playgroud)

原帖于 2022 年 11 月 6 日

简而言之,Pillow目前不支持高效图像格式 (HEIF) 文件格式。

参考:

此问题的解决方法之一是pyheif。此 Python 包具有将 HEIC 图像转换为 JPEG 图像的功能。经过这样的转换Pillow就可以从图像中读取数据了。

此格式读取问题的另一个解决方法是piexif。这是我发布的关于TIFF将文件转换为 JPEG 文件以便使用Pillow.

您还可以使用ExifTool,它可以直接读取 HEIC 文件。使用它稍微复杂一些,因为它需要使用subprocess.