【Python文本处理】基于运动路线记录GPX文件解析,心率、速度、时间、功率、踏频、海拔等参数的生成和更改
GPX文件本身其实就是坐标、海拔、时间、心率等综合性的xml文件
 如图:
 
 海拔:ele
 时间:time
 心率:heartrate
 功率:power
 踏频:cadence
 距离:distance
 一般不用距离distance 但可以根据距离求瞬时速度(前提是时间间隔均匀 最小精度不低于1s) 不过如果距离和坐标之差相差太远 则不太好确定 Strava等软件通过这个来计算瞬时速度和距离 但不用于计算赛段速度(赛段时间)
GPX文件读取和保存
def save_gpx(gpx,path):
    try:
        f=open(path, 'w', encoding="utf-8")
    except:
        f=open(path, 'a', encoding="utf-8")
    for i in gpx:
        f.write(i)
    f.close()
path="./test.gpx"
    
f=open(path, 'r', encoding="utf-8")
gpx_lines=f.readlines()
f.close()
save_gpx(gpx_lines,"./test.gpx")
传入一个GPX文件数据列表 生成GPX文件
GPX文件生成和更改
一般来说 不管是运动记录 还是路线图 trkpt后面的经纬度坐标和海拔是一定有的
 
 如图就是最基本的:
添加和更改时间
def make_time(gpx,coefficient,time_str):
    gpx_list=[]
    s=''
    first_time=time.mktime(time.strptime(time_str, "%Y-%m-%dT%H:%M:%SZ"))
    now_time=first_time
    flag=0
    for i in range(len(gpx)):
        s=gpx[i]
        if gpx[i].count('</time>') and flag==0:
            s='<time>'+time_str+'</time>\n'
            flag=1
        elif gpx[i].count('<trk>') and flag==0:
            s='<time>'+time_str+'</time>\n'+ gpx[i]
            flag=1
        elif flag==1:
            try:
                if gpx[i].count('</ele>'):
                    now_time=now_time+1*coefficient
                    change_time=time.strftime("%Y-%m-%dT%H:%M:%SZ",time.localtime(now_time))  
                    s_list=s.split("</ele>")
                    s=s_list[0]+'</ele>\n<time>'+change_time+'</time>'+s_list[1]
            except:
                pass
        gpx_list.append(s)
    return gpx_list
    
def change_time(gpx,coefficient,time_add):
    gpx_list=[]
    s=''
    first_time=0
    for i in range(len(gpx)):
        s=gpx[i]
        try:
            ti=str((gpx[i].split("<time>")[1]).split("</time>")[0])
            now_time = time.mktime(time.strptime(ti, "%Y-%m-%dT%H:%M:%SZ"))
            if first_time==0:
                first_time=now_time
                change_time=now_time+time_add
            else:
                change_time=((now_time-first_time)*coefficient)+first_time+time_add
            change_time=time.strftime("%Y-%m-%dT%H:%M:%SZ",time.localtime(change_time))  
            s=str(gpx[i].replace(ti,str(change_time)))
        except:
            pass
        gpx_list.append(s)
    return gpx_list
添加时间要传入时间间隔系数(越大 时间间隔越长 速度越慢) 以及初始时间字符串(格式"%Y-%m-%dT%H:%M:%SZ")
 更改时间同样就是要传入时间间隔系数和时间偏移量(单位s)
 
 效果
添加和更改数据
data_choice=[-5,-4,-4,-3,-3,-3,-2,-2,-2,-1,-1,-1,-1,0,0,0,0,1,1,1,1,2,2,2,3,3,3,4,4,5]
def data_list_creat(n,avg,ran):
    data_list=[]
    data_list.append(avg)
    
    min_data=avg-ran
    max_data=avg+ran
    for i in range(n-1):  
        num=data_list[i]+random.choice(data_choice) 
        if num < min_data:
            num=data_list[i]+1
        if num > max_data:
            num=data_list[i]-1
        data_list.append(num)            
        
    return data_list
以avg为中芯 ran为范围 生成n个随机数据的列表 随机取值时 从data_choice中获取
def make_data(gpx,avg,ran,keywords):
    gpx_list=[]
    data_list=[]
    num_list=[]
    s=''
    j=0
    distance=0
    for i in range(len(gpx)):
        try:
            if gpx[i].count('</time>'):
                j=j+1
                num_list.append(i)
        except:
            pass
    data_list=data_list_creat(j,avg,ran)
    j=0    
    for i in range(len(gpx)):
        s=gpx[i]   
        try:
            if i==num_list[j]: 
                s_list=s.split("</time>")                
                if keywords == "distance":
                    distance=distance+data_list[j]
                    s=s_list[0]+"</time>\n<extensions><"+keywords+">"+str(distance)+"</"+keywords+"></extensions>"+s_list[1]
                else:
                    s=s_list[0]+"</time>\n<extensions><"+keywords+">"+str(data_list[j])+"</"+keywords+"></extensions>"+s_list[1]
                j=j+1
        except:
            pass
        gpx_list.append(s)
    return gpx_list
def change_data(gpx,coefficient,keywords):
    gpx_list=[]
    s=''
    for i in range(len(gpx)):
        s=gpx[i]
        try:
            heart=str((gpx[i].split("<"+keywords+">")[1]).split("</"+keywords+">")[0])
            change=int(int(heart)*coefficient)                
            s=str(gpx[i].replace(heart,str(change)))
        except:
            pass
        gpx_list.append(s)
    return gpx_list
传入GPX列表、数据中心值和范围、数据关键词
 更改时则传入系数和关键词
 关键词有:
'''data keywords
心率 "heartrate"
踏频 "cadence"
距离 "distance"
功率 "power"
'''
其中 距离"distance"在进行make时 是累加起来的 其他的则是在时间点随机选取
 
 删除数据也要传入关键词
def clean_data(gpx,keywords):
    gpx_list=[]
    s=''
    clean_flag=0
    for i in range(len(gpx)):
        s=gpx[i]   
        if clean_flag==0:
            if gpx[i].count("<"+keywords+">"):
                clean_flag=1
                s_list=gpx[i].split("<"+keywords+">")
                if gpx[i].count("</"+keywords+">"):
                    clean_flag=0                    
                    s2_list = s_list[1].split("</"+keywords+">")
                    s_list[1]=s2_list[1]
                s=s_list[0]+s_list[1]
        else:
            if gpx[i].count("</"+keywords+">"):
                clean_flag=0
                s_list=gpx[i].split("</"+keywords+">")
                s=s_list[1]
            else:
                s=''            
        gpx_list.append(s)
    return gpx_list
更改海拔
def change_ele(gpx,coefficient,add):
    gpx_list=[]
    s=''
    for i in range(len(gpx)):
        s=gpx[i]
        try:
            el=str((gpx[i].split("<ele>")[1]).split("</ele>")[0])
            change_el=int(int(el)*coefficient+add)
            s=str(gpx[i].replace(el,str(change_el)))
        except:
            pass
        gpx_list.append(s)
    return gpx_list
def change_ele_climb(gpx,coefficient,add):
    gpx_list=[]
    s=''
    j=0
    for i in range(len(gpx)):
        s=gpx[i]
        try:
            el=str((gpx[i].split("<ele>")[1]).split("</ele>")[0])
            change_el=int(j*coefficient+add)
            j=j+1
            s=str(gpx[i].replace(el,str(change_el)))
        except:
            pass
        gpx_list.append(s)
    return gpx_list
第一个同更改时间一样 填入系数和偏移
 第二个是更改为爬坡(海拔逐渐上升)
 
更改坐标
def change_location(gpx,lat_add,lon_add):
    gpx_list=[]
    s=''
    lat_add=Decimal(str(lat_add))
    lon_add=Decimal(str(lon_add))
    for i in range(len(gpx)):
        s=gpx[i]
        if gpx[i].count('<trkpt'):
            s_list=gpx[i].split('"')
            lat=Decimal(s_list[1])+lat_add
            lon=Decimal(s_list[3])+lon_add
            s=s_list[0]+'"'+str(lat)+'"'+s_list[2]+'"'+str(lon)+'"'+s_list[4]
        gpx_list.append(s)
    return gpx_list
传入偏移即可
 
找坐标极点
def find_pole(gpx):
    south=-90
    north=90
    west=-180
    east=180
    location=[[0.0,0.0],[0.0,0.0],[0.0,0.0],[0.0,0.0]]
    for i in range(len(gpx)):
        if gpx[i].count('<trkpt'):
            s_list=gpx[i].split('"')
            lat=Decimal(s_list[1])
            lon=Decimal(s_list[3])
            if lat>south:  #最北
                south=lat
                location[0]=[lat,lon]
            if lat<north:  #最南
                north=lat
                location[1]=[lat,lon]
            if lon>west:  #最东
                west=lon
                location[2]=[lat,lon]
            if lon<east:  #最西
                east=lon
                location[3]=[lat,lon]
    return location
一般配合更改坐标使用
 
 分别输出最北点 最南点 最东点和最西点
GPX总结

# -*- coding: utf-8 -*-
"""
Created on Thu May 11 15:33:28 2023
@author: ZHOU
"""
import time
import random
from decimal import Decimal
data_choice=[-5,-4,-4,-3,-3,-3,-2,-2,-2,-1,-1,-1,-1,0,0,0,0,1,1,1,1,2,2,2,3,3,3,4,4,5]
'''data keywords
心率 "heartrate"
踏频 "cadence"
距离 "distance"
功率 "power"
'''
def find_pole(gpx):
    south=-90
    north=90
    west=-180
    east=180
    location=[[0.0,0.0],[0.0,0.0],[0.0,0.0],[0.0,0.0]]
    for i in range(len(gpx)):
        if gpx[i].count('<trkpt'):
            s_list=gpx[i].split('"')
            lat=Decimal(s_list[1])
            lon=Decimal(s_list[3])
            if lat>south:  #最北
                south=lat
                location[0]=[lat,lon]
            if lat<north:  #最南
                north=lat
                location[1]=[lat,lon]
            if lon>west:  #最东
                west=lon
                location[2]=[lat,lon]
            if lon<east:  #最西
                east=lon
                location[3]=[lat,lon]
    return location
def change_location(gpx,lat_add,lon_add):
    gpx_list=[]
    s=''
    lat_add=Decimal(str(lat_add))
    lon_add=Decimal(str(lon_add))
    for i in range(len(gpx)):
        s=gpx[i]
        if gpx[i].count('<trkpt'):
            s_list=gpx[i].split('"')
            lat=Decimal(s_list[1])+lat_add
            lon=Decimal(s_list[3])+lon_add
            s=s_list[0]+'"'+str(lat)+'"'+s_list[2]+'"'+str(lon)+'"'+s_list[4]
        gpx_list.append(s)
    return gpx_list
def change_ele(gpx,coefficient,add):
    gpx_list=[]
    s=''
    for i in range(len(gpx)):
        s=gpx[i]
        try:
            el=str((gpx[i].split("<ele>")[1]).split("</ele>")[0])
            change_el=int(int(el)*coefficient+add)
            s=str(gpx[i].replace(el,str(change_el)))
        except:
            pass
        gpx_list.append(s)
    return gpx_list
def change_ele_climb(gpx,coefficient,add):
    gpx_list=[]
    s=''
    j=0
    for i in range(len(gpx)):
        s=gpx[i]
        try:
            el=str((gpx[i].split("<ele>")[1]).split("</ele>")[0])
            change_el=int(j*coefficient+add)
            j=j+1
            s=str(gpx[i].replace(el,str(change_el)))
        except:
            pass
        gpx_list.append(s)
    return gpx_list
def data_list_creat(n,avg,ran):
    data_list=[]
    data_list.append(avg)
    
    min_data=avg-ran
    max_data=avg+ran
    for i in range(n-1):  
        num=data_list[i]+random.choice(data_choice) 
        if num < min_data:
            num=data_list[i]+1
        if num > max_data:
            num=data_list[i]-1
        data_list.append(num)            
        
    return data_list
def clean_data(gpx,keywords):
    gpx_list=[]
    s=''
    clean_flag=0
    for i in range(len(gpx)):
        s=gpx[i]   
        if clean_flag==0:
            if gpx[i].count("<"+keywords+">"):
                clean_flag=1
                s_list=gpx[i].split("<"+keywords+">")
                if gpx[i].count("</"+keywords+">"):
                    clean_flag=0                    
                    s2_list = s_list[1].split("</"+keywords+">")
                    s_list[1]=s2_list[1]
                s=s_list[0]+s_list[1]
        else:
            if gpx[i].count("</"+keywords+">"):
                clean_flag=0
                s_list=gpx[i].split("</"+keywords+">")
                s=s_list[1]
            else:
                s=''            
        gpx_list.append(s)
    return gpx_list
def make_time(gpx,coefficient,time_str):
    gpx_list=[]
    s=''
    first_time=time.mktime(time.strptime(time_str, "%Y-%m-%dT%H:%M:%SZ"))
    now_time=first_time
    flag=0
    for i in range(len(gpx)):
        s=gpx[i]
        if gpx[i].count('</time>') and flag==0:
            s='<time>'+time_str+'</time>\n'
            flag=1
        elif gpx[i].count('<trk>') and flag==0:
            s='<time>'+time_str+'</time>\n'+ gpx[i]
            flag=1
        elif flag==1:
            try:
                if gpx[i].count('</ele>'):
                    now_time=now_time+1*coefficient
                    change_time=time.strftime("%Y-%m-%dT%H:%M:%SZ",time.localtime(now_time))  
                    s_list=s.split("</ele>")
                    s=s_list[0]+'</ele>\n<time>'+change_time+'</time>'+s_list[1]
            except:
                pass
        gpx_list.append(s)
    return gpx_list
def make_data(gpx,avg,ran,keywords):
    gpx_list=[]
    data_list=[]
    num_list=[]
    s=''
    j=0
    distance=0
    for i in range(len(gpx)):
        try:
            if gpx[i].count('</time>'):
                j=j+1
                num_list.append(i)
        except:
            pass
    data_list=data_list_creat(j,avg,ran)
    j=0    
    for i in range(len(gpx)):
        s=gpx[i]   
        try:
            if i==num_list[j]: 
                s_list=s.split("</time>")                
                if keywords == "distance":
                    distance=distance+data_list[j]
                    s=s_list[0]+"</time>\n<extensions><"+keywords+">"+str(distance)+"</"+keywords+"></extensions>"+s_list[1]
                else:
                    s=s_list[0]+"</time>\n<extensions><"+keywords+">"+str(data_list[j])+"</"+keywords+"></extensions>"+s_list[1]
                j=j+1
        except:
            pass
        gpx_list.append(s)
    return gpx_list
def change_data(gpx,coefficient,keywords):
    gpx_list=[]
    s=''
    for i in range(len(gpx)):
        s=gpx[i]
        try:
            heart=str((gpx[i].split("<"+keywords+">")[1]).split("</"+keywords+">")[0])
            change=int(int(heart)*coefficient)                
            s=str(gpx[i].replace(heart,str(change)))
        except:
            pass
        gpx_list.append(s)
    return gpx_list
def change_time(gpx,coefficient,time_add):
    gpx_list=[]
    s=''
    first_time=0
    for i in range(len(gpx)):
        s=gpx[i]
        try:
            ti=str((gpx[i].split("<time>")[1]).split("</time>")[0])
            now_time = time.mktime(time.strptime(ti, "%Y-%m-%dT%H:%M:%SZ"))
            if first_time==0:
                first_time=now_time
                change_time=now_time+time_add
            else:
                change_time=((now_time-first_time)*coefficient)+first_time+time_add
            change_time=time.strftime("%Y-%m-%dT%H:%M:%SZ",time.localtime(change_time))  
            s=str(gpx[i].replace(ti,str(change_time)))
        except:
            pass
        gpx_list.append(s)
    return gpx_list
def save_gpx(gpx,path):
    try:
        f=open(path, 'w', encoding="utf-8")
    except:
        f=open(path, 'a', encoding="utf-8")
    for i in gpx:
        f.write(i)
    f.close()
    
if __name__ == '__main__':
    path="./test.gpx"
    
    f=open(path, 'r', encoding="utf-8")
    gpx_lines=f.readlines()
    f.close()    
    gpx=change_data(gpx_lines,2,"distance")
    gpx=change_time(gpx,1,-1416165)
    save_gpx(gpx,path)
    
py打包
Pyinstaller打包exe(包括打包资源文件 绝不出错版)
依赖包及其对应的版本号
PyQt5 5.10.1
 PyQt5-Qt5 5.15.2
 PyQt5-sip 12.9.0
pyinstaller 4.5.1
 pyinstaller-hooks-contrib 2021.3
Pyinstaller -F setup.py 打包exe
Pyinstaller -F -w setup.py 不带控制台的打包
Pyinstaller -F -i xx.ico setup.py 打包指定exe图标打包
打包exe参数说明:
-F:打包后只生成单个exe格式文件;
-D:默认选项,创建一个目录,包含exe文件以及大量依赖文件;
-c:默认选项,使用控制台(就是类似cmd的黑框);
-w:不使用控制台;
-p:添加搜索路径,让其找到对应的库;
-i:改变生成程序的icon图标。
如果要打包资源文件
 则需要对代码中的路径进行转换处理
 另外要注意的是 如果要打包资源文件 则py程序里面的路径要从./xxx/yy换成xxx/yy 并且进行路径转换
 但如果不打包资源文件的话 最好路径还是用作./xxx/yy 并且不进行路径转换
def get_resource_path(relative_path):
    if hasattr(sys, '_MEIPASS'):
        return os.path.join(sys._MEIPASS, relative_path)
    return os.path.join(os.path.abspath("."), relative_path)
而后再spec文件中的datas部分加入目录
 如:
a = Analysis(['cxk.py'],
             pathex=['D:\\Python Test\\cxk'],
             binaries=[],
             datas=[('root','root')],
             hiddenimports=[],
             hookspath=[],
             hooksconfig={},
             runtime_hooks=[],
             excludes=[],
             win_no_prefer_redirects=False,
             win_private_assemblies=False,
             cipher=block_cipher,
             noarchive=False)
而后直接Pyinstaller -F setup.spec即可
如果打包的文件过大则更改spec文件中的excludes 把不需要的库写进去(但是已经在环境中安装了的)就行
这些不要了的库在上一次编译时的shell里面输出
 比如:
 

 然后用pyinstaller --clean -F 某某.spec





![[VRTK4.0]设置OpenXRCameraRig](https://img-blog.csdnimg.cn/9384bdae15e64099b7befa7629efcce9.png)













