阅读文本大概需要 8 分钟。

1

目 标 场 景

女朋友会 Python 是多么可怕的一件事!

一位朋友告诉忽略了一件事,假设女朋友会 Python 的话,那岂不是要翻车?如果是这样的话,女朋友发过来的图片包含的地理位置是否真实,要打一个 Big Big 的问号了?

实际上,利用 Python 是可以非常方便地修改一张图片的地理位置。

2

编 写 代 码

在编写脚本之前,需要在虚拟环境下安装依赖库:piexif

第一步,我们读取原图的经度和纬度,然后将地理位置格式化为浮点类型。

拿到 GPS 图片坐标之后,需要转为火星坐标系数据。

def read_image(self, image_path):
    """
    开始处理图片
    exifread:读取图片属性
    :return:
    """
    exif_dict = piexif.load(image_path)

    if exif_dict['GPS']:

         # 纬度
         gps_lati_pre = exif_dict['GPS'][2]

         gps_lati = dms_to_gps(gps_lati_pre)

         # 经度
         gps_long_pre = exif_dict['GPS'][4]
         gps_long = dms_to_gps(gps_long_pre)

         # GPS坐标转为高德坐标
         lng, lat = wgs84togcj02(gps_long, gps_lati)

         print(f"原图地理位置如下\n经度:{lng}\n纬度:{lat}\n")

         return f'{lng}, {lat}'
    else:
         print(f'抱歉!这张图片不包含地理位置!')

第二步,利用高德 Web 服务 API 中的 逆地理编码 解析出原图的详细位置。

def get_address_by_location(self, location):
    """
    通过经纬度拿到地理位置
    :param location:
    :return:
    """
    params = {
       'key': self.ak,
        'location': location,
        'sig': self.sign
    }

    resp = json.loads(requests.get(url=self.url_regeo, params=params).text)

    if resp and resp.get('regeocode') and resp.get('regeocode').get('formatted_address'):
         address = resp.get('regeocode').get('formatted_address')
         print(f'原图的拍摄地址为:{address}\n')
    else:
         print('api解析地址出错,请检查ak!\n')

第三步,寻找一个要定位的地理位置

首先,输入目标城市和具体地址,利用高德 Web 服务 API 中的地理编码拿到要定位的地理位置。

def get_location_by_address(self, city, address):
    """
    通过地理位置到拿到经纬度
    地理编码:https://lbs.amap.com/api/webservice/guide/api/georegeo/
    :param address:
    :return:
    """
    params = {
            'key': self.ak,
            'city': city,
            'address': address,
            'sig': self.sign
    }

    resp = json.loads(requests.get(url=self.url_geo, params=params).text)

    # 获取坐标地址
    if resp and len(resp.get('geocodes')) >= 1 and resp.get('geocodes')[0].get('location'):
         location = resp.get('geocodes')[0].get('location')
         gps_data = location.split(',')

         # 得到经度和纬度
         gps_long = float(gps_data[0])
         gps_lati = float(gps_data[1])

         return gps_long, gps_lati
    else:
         print('api解析地址出错,请检查ak!')
         return None

第四步,修改图片的地理位置

由于 piexif 中经、纬度数据都是元组类型,因此需要将要设置的数据进行一次处理。

def gps_to_dms(gps_data):
    """
    坐标转为度、分、秒(double)
    116.397451
    http://www.gzhatu.com/du2dfm.html
    :param gps_data:
    :return:{1: b'N', 2: ((22, 1), (32, 1), (945, 100)), 3: b'E', 4: ((114, 1), (1, 1), (3445, 100))
    """
    # 度:向下取整
    gps_degree = math.floor(gps_data)

    gps_data_temp1 = (gps_data - gps_degree) * 60

    # 分
    gps_minute = math.floor(gps_data_temp1)

    gps_data_temp2 = gps_data_temp1 - gps_minute

    # 秒,取小数点后4位
    gps_second = round(gps_data_temp2 * 60, 2)

    # 注意:秒必须转换为整形
    result = ((gps_degree, 1), (gps_minute, 1), (int(gps_second * 100), 100))

    return result

最后,将正确格式的经纬度数据写入到图片中去。

def write_image(self, image_path, gps_long, gps_lati):
    """
    修改文件夹下所有文件的属性
    :param image_path: 文件夹路径
    :return:
    """
    # 读取图片
    img = Image.open(image_path)

    try:
        exif_dict = piexif.load(img.info['exif'])
    except:
        print('加载文件地理位置异常!')
        return

    # 修改地理位置
    # GPS GPSLatitudeRef:N
    # GPS GPSLatitude:[22, 32, 189/20]
    # GPS GPSLongitudeRef:E
    # GPS GPSLongitude:[114, 1, 689/20]
    exif_dict['GPS'][2] = gps_to_dms(gps_lati)
    exif_dict['GPS'][4] = gps_to_dms(gps_long)

    exif_bytes = piexif.dump(exif_dict)

    # 写入到新的图片中去
    img.save(image_path, 'jpeg', exif=exif_bytes)

3

结 果 结 论

通过上面的 4 步操作,能将任意地理位置写入到图片中。

除了地理位置,图片的元数据,包含:拍摄器材、拍摄时间、拍摄参数都能进行二次修改

女朋友会 Python 是多么可怕的一件事!的更多相关文章

  1. Python运维开发基础08-文件基础【转】

    一,文件的其他打开模式 "+"表示可以同时读写某个文件: r+,可读写文件(可读:可写:可追加) w+,写读(不常用) a+,同a(不常用 "U"表示在读取时, ...

  2. Python运维开发基础07-文件基础【转】

    一,文件的基础操作 对文件操作的流程 [x] :打开文件,得到文件句柄并赋值给一个变量 [x] :通过句柄对文件进行操作 [x] :关闭文件 创建初始操作模板文件 [root@localhost sc ...

  3. 关于我用python表白成功这件事【表白成功】

    520,并非情人所属, 我们可以表白万物, 不管什么时候, 这都是一个特别的日子, 今天,我要表白所有, 心里有我的人! 在这个充满幸福的日子里, 我要把最美好的祝福, 送给心里有我的每一个人: 祝愿 ...

  4. Selenium2+python自动化25-js处理日历控件(修改readonly属性)

    前言 日历控件是web网站上经常会遇到的一个场景,有些输入框是可以直接输入日期的,有些不能,以我们经常抢票的12306网站为例,详细讲解如何解决日历控件为readonly属性的问题. 基本思路:先用j ...

  5. python通过win32api轻松获取控件的属性值

    1.如何利用句柄操作windows窗体 首先,获得窗体的句柄  win32api.FindWindows() 第二,获得窗体中控件的id号,spy++ 第三,根据控件的ID获得控件的句柄(hwnd)  ...

  6. Python核心编程--学习笔记--9--文件和输入输出

    本章将深入介绍Python的文件处理和相关输入输出能力,包括:文件对象(以及它的内建函数.内建方法和属性),标准文件,文件系统的访问方法,文件执行,最后简要涉及持久存储和标准库中与文件有关的模块. 1 ...

  7. Python核心编程--学习笔记--8--条件与循环

    本章讲述if.while.for以及与他们搭配的else.elif.break.continue.pass等语句. 1 if语句 语法:三部分——关键字if.条件表达式.代码块.(记住冒号) if c ...

  8. PYTHON线程知识再研习E---条件变量同步Condition

    Python提供的Condition对象提供了对复杂线程同步问题的支持.Condition被称为条件变量,除了提供与Lock类似的 acquire和release方法外,还提供了wait和notify ...

  9. 利用python的爬虫技术爬去糗事百科的段子

    初次学习爬虫技术,在知乎上看了如何爬去糗事百科的段子,于是打算自己也做一个. 实现目标:1,爬取到糗事百科的段子 2,实现每次爬去一个段子,每按一次回车爬取到下一页 技术实现:基于python的实现, ...

随机推荐

  1. 【CSS】381- 提升你的CSS选择器技巧

    我已经使用CSS多年了,但直到最近我才深入研究了一下CSS选择器. 我为什么要这样做呢?我们都知道选择器,但麻烦的是随着时间的推移,很容易习惯于在每个项目中使用相同的可信任选择器来实现你需要做的事情. ...

  2. 《JavaScript 模式》知识点小抄本(下)

    介绍 最近开始给自己每周订个学习任务,学习结果反馈为一篇文章的输出,做好学习记录. 这一周(02.25-03.03)我定的目标是<JavaScript 模式>的第七章学习一遍,学习结果的反 ...

  3. python数据结构——单向链表

    链表 ( Linked List ) 定义:由许多相同数据类型的数据项按照特定顺序排列而成的线性表. 特点:各个数据在计算机中是随机存放且不连续. 优点:数据的增删改查都很方便,当有新的数据加入的时候 ...

  4. 从多谐振荡器详细解析到555定时器基本电路(控制LED闪烁)

    在学期末,笔者参加了学校的电工实习,前六天做都很快,但是今天要做一个关于555多谐振荡器的LED闪烁电路,由于笔者没有提前准备,导致今天就算把电路搭建出来也不懂具体原理,耗费了不少时间,所以我打算专门 ...

  5. 一线大厂Java面试必问的2大类Tomcat调优

    一.前言 最近整理了 Tomcat 调优这块,基本上面试必问,于是就花了点时间去搜集一下 Tomcat 调优都调了些什么,先记录一下调优手段,更多详细的原理和实现以后用到时候再来补充记录,下面就来介绍 ...

  6. webpack打包出现WARNING in configuration The 'mode' option has not been set, webpack will fallback to 'production' for this value. 错误

    打包运行的时候出现以下错误 WARNING in configurationThe 'mode' option has not been set, webpack will fallback to ' ...

  7. 表达式和运算符知识总结(js)

    文章目录: 一. 表达式和语句的区别 二. 自增自减运算符的运算规则 一. 表达式和语句的区别 表达式(expression)是JavaScript中的一个短语,JavaScript解释器会将其计算( ...

  8. 蝉知CMS5.6反射型XSS审计复现

    0x00 源起 最近在深入学习反射XSS时遇到蝉知CMS5.6反射型XSS这个案列,乍一看网上的漏洞介绍少之又少,也没有详细的审计复现流程.虽然是17年的漏洞了,不巧本人正是一个喜欢钻研的人.这个CM ...

  9. 脚本批量执行Redis命令

    1.将命令写在文件中 数据量比较大的话,建议用程序去生成文件.例如: List<String> planIdList = planDao.findAll().parallelStream( ...

  10. inux 网络监控分析

    一.sar -n:查看网卡流量 -n 参数,他有6个不同的开关:DEV | EDEV | NFS | NFSD | SOCK | ALL .DEV显示网络接口信息,EDEV显示关于网络错误的统计数据, ...