阅读文本大概需要 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. JSP请求是如何被处理的?jsp的执行原理

    客户端通过浏览器发送jsp请求,服务器端接受到请求后,判断是否是第一次请求该页面,或者该页面是否改变,若是,服务器将jsp页面翻译为servlet,jvm将servlet编译为.class文件,字节码 ...

  2. 使用Pycharm轻轻松松脱下git版本控制高大上的外衣

    一.思考❓❔ 1.git操作难吗? git操作命令繁杂 需求复杂场景, 使用不易 原理深邃,对初学者来说是有难度的 2.那么难,还要学吗? 作为IT行业从业者(搬砖小工),不会Git?滚,出去~~~ ...

  3. AQS系列(五)- CountDownLatch的使用及原理

    前言 前面四节学完了AQS最难的两种重入锁应用,下面两节进入实战学习,看看JUC包中其他的工具类是如何运用AQS实现特定功能的.今天一起看一下CountDownLatch. CountDownLatc ...

  4. UWP 利用Windows.UI.Composition实现简单的放大🔍效果

    看一下效果先 我这里实现了鼠标进入和退出的效果,当然也可以添加其他的事件,比如获得焦点和失去焦点的. 先随便写一个xaml布局,一个Grid,里面一张图片. <Grid x:Name=" ...

  5. 去掉 Idea 中注入 Mapper 警告的方法

    使用 Idea 的时候,自动装配 Mybatis 的 mapper.会一直出现红色波浪线的警告.看着难受.下面提供几种方式 方式一 为 @Autowired 注解设置required = false ...

  6. VS2019添加Link to SQL类之后,LinqDataSource配置数据源没有可选项

    原创. 环境:visual studio 2019 ,.net framwork 4.0 问题: 1.新建Linq to SQL类后(.dbml文件),从服务器资源管理面板中数据库连接中,把表Mill ...

  7. javascript中引用传递的问题如何解决

    我们有时候会向一个方法中传入一个参数,并且对这个参数做一些处理的操作: 但是因为是引用传递,处理过后会对原有的对象造成修改,无法进行反复使用. 如例子: 两次打印的结果一模一样.这样下一个方法在继续使 ...

  8. Linux 桥接网络不自动分配IP的问题

    之前遇到过好多次,知道什么原因就是忘了命令怎么敲,还要去搜索,写一遍加强下记忆,并总结下. 情况一 :网卡冲突问题 1 , 网卡问题 有安装过oracle VM VirtualBox 的,会和VMwa ...

  9. 监听属性watch

    immediate.deep data() { return { firstName: "刘", lastName: "XX", fullName: " ...

  10. 基于V6的中移动物联测试例子,当前测试还挺稳定

    下载: 链接:https://pan.baidu.com/s/1Gz8mEffDGXNSK8lIsAIUEg   提取码:2sur 测试步骤看此贴,跟V7开发板是一样的: 基于H7的中移动物联例子以及 ...