阅读文本大概需要 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. jmeter微信公众号接口测试实例

    线程组 HTTP Cookie 管理器 HTTP 请求默认值 用户定义的变量 察看结果树 HTTP请求 响应断言 正则表达式提取器 线程组 HTTP Cookie 管理器 HTTP 请求默认值 用户定 ...

  2. Python流程控制之循环结构

    目录 while循环 for循环 嵌套循环 break.continue.pass 练习 当出现有规律或者是重复的事情就可以使用循环. 1.循环变量初始化 2.循环条件 3.循环体 4.改变循环变量 ...

  3. 【Java Web开发学习】Spring MVC添加自定义Servlet、Filter、Listener

    [Java Web开发学习]Spring MVC添加自定义Servlet.Filter.Listener 转载:https://www.cnblogs.com/yangchongxing/p/9968 ...

  4. CSRF与auth模块

    目录 一.模拟实现中间件的编程思想 (一)impotlib模块 (二)实现功能的配置使用 二.跨站请求伪造CSRF (一)由来 (二)form表单的CSRF (三)ajax中的CSRF (1)通过da ...

  5. Internet History,Technology,and Security -Transport Control Protocol(TCP)(Week6)

    Week6 Technology: Transport Control Protocol(TCP) Welcome to Week 6 of IHTS. We are in our second we ...

  6. JS---案例:移动元素,封装动画函数

    案例:移动元素,封装动画函数 1. div要移动,要脱离文档流---position:absolute 2. 如果样式的代码是在style的标签中设置,外面是获取不到 3. 如果样式的代码是在styl ...

  7. 错题本:ConstraintLayout 不能正常显示

    理想效果: 实际效果: 原因:因为文件中一个控件的约束属性写错了 这个属性是 app:layout_constraintLeft_toLeftOf="@id/oa_setting_group ...

  8. mysql中msvcr120.dll文件丢失问题

    安装VC++2013 若是以上方法不能解决,需要下载安装VC++2013,这是微软官网的链接 https://www.microsoft.com/zh-cn/download/confirmation ...

  9. Python类中的self的作用

    Python编写类的时候,每个函数第一个参数都是self.后来对Python越来越熟悉,再回头看self的概念,慢慢就明白了. 谷歌上有一段解释很到位,贴出来给大家: self represents ...

  10. 防止sql注入的最好方式

    避免 SQL injection 攻击的传统方法之一是,把它作为一个输入合法性检查的问题来处理,只接受列在白名单中的字符,或者识别并避免那些列在黑名单中的恶意数据.白名单方法是一种非常有效方法,它可以 ...