-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrobot_class.py
128 lines (112 loc) · 5.29 KB
/
robot_class.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
from math import *
import random
# ------------------------------------- ###
# Below, is the robot class
#
# This robot lives in 2D, x-y space, and its motion is
# pointed in a random direction, initially.
# It moves in a straight line until it comes close to a wall
# at which point it stops.
#
# For measurements, it senses the x- and y-distance
# to landmarks. This is different from range and bearing as
# commonly studied in the literature, but this makes it much
# easier to implement the essentials of SLAM without
# cluttered math.
#
class robot:
# --------
# init:
# creates a robot with the specified parameters and initializes
# the location (self.x, self.y) to the center of the world
#
def __init__(self, world_size=100.0, measurement_range=30.0,
motion_noise=1.0, measurement_noise=1.0):
self.measurement_noise = 0.0
self.world_size = world_size
self.measurement_range = measurement_range
self.x = world_size / 2.0
self.y = world_size / 2.0
self.motion_noise = motion_noise
self.measurement_noise = measurement_noise
self.landmarks = []
self.num_landmarks = 0
# returns a positive, random float
def rand(self):
return random.random() * 2.0 - 1.0
# --------
# move: attempts to move robot by dx, dy. If outside world
# boundary, then the move does nothing and instead returns failure
#
def move(self, dx, dy):
x = self.x + dx + self.rand() * self.motion_noise
y = self.y + dy + self.rand() * self.motion_noise
if x < 0.0 or x > self.world_size or y < 0.0 or y > self.world_size:
return False
else:
self.x = x
self.y = y
return True
# --------
# sense: returns x- and y- distances to landmarks within visibility range
# because not all landmarks may be in this range, the list of measurements
# is of variable length. Set measurement_range to -1 if you want all
# landmarks to be visible at all times
#
# TODO: paste your complete the sense function, here
# make sure the indentation of the code is correct
def sense(self):
''' This function does not take in any parameters, instead it references internal variables
(such as self.landamrks) to measure the distance between the robot and any landmarks
that the robot can see (that are within its measurement range).
This function returns a list of landmark indices, and the measured distances (dx, dy)
between the robot's position and said landmarks.
This function should account for measurement_noise and measurement_range.
One item in the returned list should be in the form: [landmark_index, dx, dy].
'''
measurements = []
# TODO: iterate through all of the landmarks in a world
for i, lm in zip(range(len(self.landmarks)), self.landmarks):
# TODO: For each landmark
# 1. compute dx and dy, the distances between the robot and the landmark
# 2. account for measurement noise by *adding* a noise component to dx and dy
# - The noise component should be a random value between [-1.0, 1.0)*measurement_noise
# - Feel free to use the function self.rand() to help calculate this noise component
# - It may help to reference the `move` function for noise calculation
# 3. If either of the distances, dx or dy, fall outside of the internal var, measurement_range
# then we cannot record them; if they do fall in the range, then add them to the measurements list
# as list.append([index, dx, dy]), this format is important for data creation done later
# print("lm: {}".format(lm))
noise = self.rand2()
# print("Noise: {}".format(noise))
dx = lm[0] - self.x + noise
noise = self.rand2()
# print("Noise: {}".format(noise))
dy = lm[1] - self.y + noise
# print(i, dx,dy)
if dx < 0: dx = dx * -1
if dy < 0: dy = dy * -1
# print(i, dx,dy)
if dx < 0.0 or dx > self.world_size or dy < 0.0 or dy > self.world_size:
continue
if dx > self.measurement_range or dy > self.measurement_range:
continue
measurements.append([i,dx,dy])
# TODO: return the final, complete list of measurements
return measurements
def rand2(self):
return random.uniform(-1.0, 1.0) * self.measurement_noise
# --------
# make_landmarks:
# make random landmarks located in the world
#
def make_landmarks(self, num_landmarks):
self.landmarks = []
for i in range(num_landmarks):
self.landmarks.append([round(random.random() * self.world_size),
round(random.random() * self.world_size)])
self.num_landmarks = num_landmarks
# called when print(robot) is called; prints the robot's location
def __repr__(self):
return 'Robot: [x=%.5f y=%.5f]' % (self.x, self.y)
# END robot class #######