女朋友会 Python 是多么可怕的一件事!
阅读文本大概需要 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 是多么可怕的一件事!的更多相关文章
- Python运维开发基础08-文件基础【转】
一,文件的其他打开模式 "+"表示可以同时读写某个文件: r+,可读写文件(可读:可写:可追加) w+,写读(不常用) a+,同a(不常用 "U"表示在读取时, ...
- Python运维开发基础07-文件基础【转】
一,文件的基础操作 对文件操作的流程 [x] :打开文件,得到文件句柄并赋值给一个变量 [x] :通过句柄对文件进行操作 [x] :关闭文件 创建初始操作模板文件 [root@localhost sc ...
- 关于我用python表白成功这件事【表白成功】
520,并非情人所属, 我们可以表白万物, 不管什么时候, 这都是一个特别的日子, 今天,我要表白所有, 心里有我的人! 在这个充满幸福的日子里, 我要把最美好的祝福, 送给心里有我的每一个人: 祝愿 ...
- Selenium2+python自动化25-js处理日历控件(修改readonly属性)
前言 日历控件是web网站上经常会遇到的一个场景,有些输入框是可以直接输入日期的,有些不能,以我们经常抢票的12306网站为例,详细讲解如何解决日历控件为readonly属性的问题. 基本思路:先用j ...
- python通过win32api轻松获取控件的属性值
1.如何利用句柄操作windows窗体 首先,获得窗体的句柄 win32api.FindWindows() 第二,获得窗体中控件的id号,spy++ 第三,根据控件的ID获得控件的句柄(hwnd) ...
- Python核心编程--学习笔记--9--文件和输入输出
本章将深入介绍Python的文件处理和相关输入输出能力,包括:文件对象(以及它的内建函数.内建方法和属性),标准文件,文件系统的访问方法,文件执行,最后简要涉及持久存储和标准库中与文件有关的模块. 1 ...
- Python核心编程--学习笔记--8--条件与循环
本章讲述if.while.for以及与他们搭配的else.elif.break.continue.pass等语句. 1 if语句 语法:三部分——关键字if.条件表达式.代码块.(记住冒号) if c ...
- PYTHON线程知识再研习E---条件变量同步Condition
Python提供的Condition对象提供了对复杂线程同步问题的支持.Condition被称为条件变量,除了提供与Lock类似的 acquire和release方法外,还提供了wait和notify ...
- 利用python的爬虫技术爬去糗事百科的段子
初次学习爬虫技术,在知乎上看了如何爬去糗事百科的段子,于是打算自己也做一个. 实现目标:1,爬取到糗事百科的段子 2,实现每次爬去一个段子,每按一次回车爬取到下一页 技术实现:基于python的实现, ...
随机推荐
- jmeter微信公众号接口测试实例
线程组 HTTP Cookie 管理器 HTTP 请求默认值 用户定义的变量 察看结果树 HTTP请求 响应断言 正则表达式提取器 线程组 HTTP Cookie 管理器 HTTP 请求默认值 用户定 ...
- Python流程控制之循环结构
目录 while循环 for循环 嵌套循环 break.continue.pass 练习 当出现有规律或者是重复的事情就可以使用循环. 1.循环变量初始化 2.循环条件 3.循环体 4.改变循环变量 ...
- 【Java Web开发学习】Spring MVC添加自定义Servlet、Filter、Listener
[Java Web开发学习]Spring MVC添加自定义Servlet.Filter.Listener 转载:https://www.cnblogs.com/yangchongxing/p/9968 ...
- CSRF与auth模块
目录 一.模拟实现中间件的编程思想 (一)impotlib模块 (二)实现功能的配置使用 二.跨站请求伪造CSRF (一)由来 (二)form表单的CSRF (三)ajax中的CSRF (1)通过da ...
- 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 ...
- JS---案例:移动元素,封装动画函数
案例:移动元素,封装动画函数 1. div要移动,要脱离文档流---position:absolute 2. 如果样式的代码是在style的标签中设置,外面是获取不到 3. 如果样式的代码是在styl ...
- 错题本:ConstraintLayout 不能正常显示
理想效果: 实际效果: 原因:因为文件中一个控件的约束属性写错了 这个属性是 app:layout_constraintLeft_toLeftOf="@id/oa_setting_group ...
- mysql中msvcr120.dll文件丢失问题
安装VC++2013 若是以上方法不能解决,需要下载安装VC++2013,这是微软官网的链接 https://www.microsoft.com/zh-cn/download/confirmation ...
- Python类中的self的作用
Python编写类的时候,每个函数第一个参数都是self.后来对Python越来越熟悉,再回头看self的概念,慢慢就明白了. 谷歌上有一段解释很到位,贴出来给大家: self represents ...
- 防止sql注入的最好方式
避免 SQL injection 攻击的传统方法之一是,把它作为一个输入合法性检查的问题来处理,只接受列在白名单中的字符,或者识别并避免那些列在黑名单中的恶意数据.白名单方法是一种非常有效方法,它可以 ...