-
Notifications
You must be signed in to change notification settings - Fork 39
/
Copy pathrun.py
155 lines (135 loc) · 5.24 KB
/
run.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
"""
run.py
automatically run the route
"""
"""修正坐标误差,百度取点使用 BD-09 坐标系,iOS使用 WGS-09 坐标系,进行转换"""
import math
import time
import random
from geopy.distance import geodesic
from driver import location
def bd09Towgs84(position):
wgs_p = {}
x_pi = 3.14159265358979324 * 3000.0 / 180.0
pi = 3.141592653589793238462643383 # π
a = 6378245.0 # 长半轴
ee = 0.00669342162296594323 # 偏心率平方
def transform_lat(x, y):
ret = -100.0 + 2.0 * x + 3.0 * y + 0.2 * y * y + 0.1 * x * y + 0.2 * math.sqrt(abs(x))
ret += (20.0 * math.sin(6.0 * x * pi) + 20.0 * math.sin(2.0 * x * pi)) * 2.0 / 3.0
ret += (20.0 * math.sin(y * pi) + 40.0 * math.sin(y / 3.0 * pi)) * 2.0 / 3.0
ret += (160.0 * math.sin(y / 12.0 * pi) + 320 * math.sin(y * pi / 30.0)) * 2.0 / 3.0
return ret
def transform_lon(x, y):
ret = 300.0 + x + 2.0 * y + 0.1 * x * x + 0.1 * x * y + 0.1 * math.sqrt(abs(x))
ret += (20.0 * math.sin(6.0 * x * pi) + 20.0 * math.sin(2.0 * x * pi)) * 2.0 / 3.0
ret += (20.0 * math.sin(x * pi) + 40.0 * math.sin(x / 3.0 * pi)) * 2.0 / 3.0
ret += (150.0 * math.sin(x / 12.0 * pi) + 300.0 * math.sin(x / 30.0 * pi)) * 2.0 / 3.0
return ret
x = position['lng'] - 0.0065
y = position['lat'] - 0.006
z = math.sqrt(x * x + y * y) - 0.00002 * math.sin(y * x_pi)
theta = math.atan2(y, x) - 0.000003 * math.cos(x * x_pi)
gcj_lng = z * math.cos(theta)
gcj_lat = z * math.sin(theta)
d_lat = transform_lat(gcj_lng - 105.0, gcj_lat - 35.0)
d_lng = transform_lon(gcj_lng - 105.0, gcj_lat - 35.0)
rad_lat = gcj_lat / 180.0 * pi
magic = math.sin(rad_lat)
magic = 1 - ee * magic * magic
sqrt_magic = math.sqrt(magic)
d_lng = (d_lng * 180.0) / (a / sqrt_magic * math.cos(rad_lat) * pi)
d_lat = (d_lat * 180.0) / (a * (1 - ee) / (magic * sqrt_magic) * pi)
wgs_p["lat"] = gcj_lat * 2 - gcj_lat - d_lat
wgs_p["lng"] = gcj_lng * 2 - gcj_lng - d_lng
return wgs_p
# get the ditance according to the latitude and longitude
def geodistance(p1, p2):
return geodesic((p1["lat"],p1["lng"]),(p2["lat"],p2["lng"])).m
def smooth(start, end, i):
import math
i = (i-start)/(end-start)*math.pi
return math.sin(i)**2
def randLoc(loc: list, d=0.000025, n=5):
import random
import time
import math
# deepcopy loc
result = []
for i in loc:
result.append(i.copy())
center = {"lat": 0, "lng": 0}
for i in result:
center["lat"] += i["lat"]
center["lng"] += i["lng"]
center["lat"] /= len(result)
center["lng"] /= len(result)
random.seed(time.time())
for i in range(n):
start = int(i*len(result)/n)
end = int((i+1)*len(result)/n)
offset = (2*random.random()-1) * d
for j in range(start, end):
distance = math.sqrt(
(result[j]["lat"]-center["lat"])**2 + (result[j]["lng"]-center["lng"])**2
)
if 0 == distance:
continue
result[j]["lat"] += (result[j]["lat"]-center["lat"])/distance*offset*smooth(start, end, j)
result[j]["lng"] += (result[j]["lng"]-center["lng"])/distance*offset*smooth(start, end, j)
start = int(i*len(result)/n)
end = len(result)
offset = (2*random.random()-1) * d
for j in range(start, end):
distance = math.sqrt(
(result[j]["lat"]-center["lat"])**2 + (result[j]["lng"]-center["lng"])**2
)
if 0 == distance:
continue
result[j]["lat"] += (result[j]["lat"]-center["lat"])/distance*offset*smooth(start, end, j)
result[j]["lng"] += (result[j]["lng"]-center["lng"])/distance*offset*smooth(start, end, j)
return result
def fixLockT(loc: list, v, dt):
fixedLoc = []
t = 0
T = []
T.append(geodistance(loc[1],loc[0])/v)
a = loc[0].copy()
b = loc[1].copy()
j = 0
while t < T[0]:
xa = a["lat"] + j*(b["lat"]-a["lat"])/(max(1, int(T[0]/dt)))
xb = a["lng"] + j*(b["lng"]-a["lng"])/(max(1, int(T[0]/dt)))
fixedLoc.append({"lat": xa, "lng": xb})
j += 1
t += dt
for i in range(1, len(loc)):
T.append(geodistance(loc[(i+1)%len(loc)],loc[i])/v + T[-1])
a = loc[i].copy()
b = loc[(i+1)%len(loc)].copy()
j = 0
while t < T[i]:
xa = a["lat"] + j*(b["lat"]-a["lat"])/(max(1, int((T[i]-T[i-1])/dt)))
xb = a["lng"] + j*(b["lng"]-a["lng"])/(max(1, int((T[i]-T[i-1])/dt)))
fixedLoc.append({"lat": xa, "lng": xb})
j += 1
t += dt
return fixedLoc
def run1(dvt, loc: list, v, dt=0.2):
fixedLoc = fixLockT(loc, v, dt)
nList = (5, 6, 7, 8, 9)
n = nList[random.randint(0, len(nList)-1)]
fixedLoc = randLoc(fixedLoc, n=n) # a path will be divided into n parts for random route
clock = time.time()
for i in fixedLoc:
# utils.setLoc(bd09Towgs84(i))
location.set_location(dvt, **bd09Towgs84(i))
while time.time()-clock < dt:
pass
clock = time.time()
def run(dvt, loc: list, v, d=15):
random.seed(time.time())
while True:
vRand = 1000/(1000/v-(2*random.random()-1)*d)
run1(dvt, loc, vRand)
print("跑完一圈了")