蓝桥ROS之f1tenth简单PID沿墙跑起来(Python)

举报
zhangrelay 发表于 2022/05/03 22:08:22 2022/05/03
【摘要】 在此案例前需完成: ☞ ​​​​​​蓝桥ROS之f1tenth案例学习与调试(成功)         遥控肯定不过瘾,那么如何用一个PID程序使小车自动沿墙跑呢??? 公式如上…… 跑一跑看看?   阅读pdf文档: python程序模板: #!/usr...

在此案例前需完成:

☞ ​​​​​​蓝桥ROS之f1tenth案例学习与调试(成功)

 

 


 

 

遥控肯定不过瘾,那么如何用一个PID程序使小车自动沿墙跑呢???

公式如上……

跑一跑看看?

 

阅读pdf文档:

python程序模板:


  
  1. #!/usr/bin/env python
  2. from __future__ import print_function
  3. import sys
  4. import math
  5. import numpy as np
  6. #ROS Imports
  7. import rospy
  8. from sensor_msgs.msg import Image, LaserScan
  9. from ackermann_msgs.msg import AckermannDriveStamped, AckermannDrive
  10. #PID CONTROL PARAMS
  11. kp = #TODO
  12. kd = #TODO
  13. ki = #TODO
  14. servo_offset = 0.0
  15. prev_error = 0.0
  16. error = 0.0
  17. integral = 0.0
  18. #WALL FOLLOW PARAMS
  19. ANGLE_RANGE = 270 # Hokuyo 10LX has 270 degrees scan
  20. DESIRED_DISTANCE_RIGHT = 0.9 # meters
  21. DESIRED_DISTANCE_LEFT = 0.55
  22. VELOCITY = 2.00 # meters per second
  23. CAR_LENGTH = 0.50 # Traxxas Rally is 20 inches or 0.5 meters
  24. class WallFollow:
  25. """ Implement Wall Following on the car
  26. """
  27. def __init__(self):
  28. #Topics & Subs, Pubs
  29. lidarscan_topic = '/scan'
  30. drive_topic = '/nav'
  31. self.lidar_sub = #TODO: Subscribe to LIDAR
  32. self.drive_pub = #TODO: Publish to drive
  33. def getRange(self, data, angle):
  34. # data: single message from topic /scan
  35. # angle: between -45 to 225 degrees, where 0 degrees is directly to the right
  36. # Outputs length in meters to object with angle in lidar scan field of view
  37. #make sure to take care of nans etc.
  38. #TODO: implement
  39. return 0.0
  40. def pid_control(self, error, velocity):
  41. global integral
  42. global prev_error
  43. global kp
  44. global ki
  45. global kd
  46. angle = 0.0
  47. #TODO: Use kp, ki & kd to implement a PID controller for
  48. drive_msg = AckermannDriveStamped()
  49. drive_msg.header.stamp = rospy.Time.now()
  50. drive_msg.header.frame_id = "laser"
  51. drive_msg.drive.steering_angle = angle
  52. drive_msg.drive.speed = velocity
  53. self.drive_pub.publish(drive_msg)
  54. def followLeft(self, data, leftDist):
  55. #Follow left wall as per the algorithm
  56. #TODO:implement
  57. return 0.0
  58. def lidar_callback(self, data):
  59. """
  60. """
  61. error = 0.0 #TODO: replace with error returned by followLeft
  62. #send error to pid_control
  63. self.pid_control(error, VELOCITY)
  64. def main(args):
  65. rospy.init_node("WallFollow_node", anonymous=True)
  66. wf = WallFollow()
  67. rospy.sleep(0.1)
  68. rospy.spin()
  69. if __name__=='__main__':
  70. main(sys.argv)

 参考程序:

scan.py


  
  1. import rospy
  2. from sensor_msgs.msg import LaserScan
  3. def callback(data):
  4. print(data.ranges[270])
  5. rospy.init_node("scan_test", anonymous=False)
  6. sub = rospy.Subscriber("/scan", LaserScan, callback)
  7. rospy.spin()

dist.py 


  
  1. import rospy
  2. import math
  3. from sensor_msgs.msg import LaserScan
  4. from race.msg import pid_input
  5. angle_range = 360 # sensor angle range of the lidar
  6. car_length = 1.5 # projection distance we project car forward.
  7. vel = 1.5 # used for pid_vel (not much use).
  8. error = 0.0
  9. dist_from_wall = 0.8
  10. pub = rospy.Publisher('error', pid_input, queue_size=10)
  11. def getRange(data,theta):
  12. angle_range = data.angle_max - data.angle_min
  13. angle_increment = data.angle_increment
  14. scan_range = []
  15. rad2deg_factor = 57.296
  16. angle_range *= rad2deg_factor
  17. angle_increment *= rad2deg_factor
  18. for element in data.ranges:
  19. if math.isnan(element) or math.isinf(element):
  20. element = 100
  21. scan_range.append(element)
  22. index = round(theta / angle_increment)
  23. return scan_range[index]
  24. def callback(data):
  25. theta = 55
  26. left_dist = float(getRange(data, 270))
  27. a = getRange(data,(90 + theta)) # Ray at degree theta from right_dist
  28. right_dist = getRange(data,90) # Ray perpendicular to right side of car
  29. theta_r = math.radians(theta)
  30. # dist_from_wall = (left_dist + right_dist)/2 # keep in middle
  31. # Calculating the deviation of steering(alpha)
  32. alpha = math.atan( (a * math.cos(theta_r) - right_dist)/ a * math.sin(theta_r) )
  33. AB = right_dist * math.cos(alpha) # Present Position
  34. CD = AB + car_length * math.sin(alpha) # projection in Future Position
  35. error = dist_from_wall - CD # total error
  36. # print('error: ', error) #Testing
  37. # Sending PID error to Control
  38. msg = pid_input()
  39. msg.pid_error = error
  40. msg.pid_vel = vel
  41. pub.publish(msg)
  42. if __name__ == '__main__':
  43. print("Laser node started")
  44. rospy.init_node('dist_finder',anonymous = True)
  45. rospy.Subscriber("scan",LaserScan,callback)
  46. rospy.spin()

control.py


  
  1. import rospy
  2. from race.msg import pid_input
  3. from ackermann_msgs.msg import AckermannDriveStamped
  4. kp = 0.7 #45
  5. kd = 0.00125#0.001 #0.09
  6. ki = 0#0.5
  7. kp_vel = 6 #9 is using keep in middle
  8. kd_vel = 0.0001
  9. ki_error = 0
  10. prev_error = 0.0
  11. vel_input = 2.5 # base velocity
  12. angle = 0.0 # initial steering angle
  13. pub = rospy.Publisher("/drive", AckermannDriveStamped, queue_size=5)
  14. def control(data):
  15. global prev_error
  16. global vel_input
  17. global kp
  18. global kp_vel
  19. global kd
  20. global kd_vel
  21. global ki
  22. global angle
  23. e = data.pid_error
  24. # Calculating deviation for lateral deviation from path
  25. kp_error = kp * e
  26. kd_error = kd * (e - prev_error)
  27. # Calculating error for velocity
  28. kp_vel_er = kp_vel * e
  29. kd_vel_er = kd * (e - prev_error)
  30. # ki_error = ki_error + (ki * e)
  31. vel_error = kp_vel_er + kd_vel_er
  32. pid_error = kp_error + kd_error
  33. min_angle=-20
  34. max_angle= 20
  35. # Heigher error results in lower velocity
  36. # while lower error results in heigher velocity
  37. velo = vel_input + 1/(abs(vel_error))
  38. #corrected steering angle
  39. angle = pid_error
  40. #print("raw velo:", velo) # Testing
  41. #Speed limit
  42. if velo > 15 :
  43. velo = 10
  44. # Filtering steering angle for Out-of-Range values
  45. if angle < min_angle:
  46. angle = min_angle
  47. elif angle > max_angle:
  48. angle = max_angle
  49. # print("filtered angle :" , angle) # Testing
  50. # Sending Drive information to Car
  51. msg = AckermannDriveStamped()
  52. msg.header.stamp = rospy.Time.now()
  53. msg.header.frame_id = ''
  54. msg.drive.speed = velo
  55. msg.drive.steering_angle = angle
  56. pub.publish(msg)
  57. if __name__ == '__main__':
  58. print("Starting control...")
  59. rospy.init_node('pid_controller', anonymous=True)
  60. rospy.Subscriber("error", pid_input, control)
  61. rospy.spin()

 centre_wall_follow.py 


  
  1. import rospy
  2. import math
  3. from sensor_msgs.msg import LaserScan
  4. from race.msg import pid_input
  5. angle_range = 360 # sensor angle range of the lidar
  6. car_length = 1.5 # projection distance we project car forward.
  7. vel = 1.0 # used for pid_vel (not much use).
  8. error = 0.0
  9. pub = rospy.Publisher('error', pid_input, queue_size=10)
  10. def getRange(data,theta):
  11. angle_range = data.angle_max - data.angle_min
  12. angle_increment = data.angle_increment
  13. scan_range = []
  14. rad2deg_factor = 57.296
  15. angle_range *= rad2deg_factor
  16. angle_increment *= rad2deg_factor
  17. for element in data.ranges:
  18. if math.isnan(element) or math.isinf(element):
  19. element = 100
  20. scan_range.append(element)
  21. index = round(theta / angle_increment)
  22. return scan_range[index]
  23. def callback(data):
  24. dist_in_front = getRange(data, 180)
  25. theta = 30
  26. left_dist = getRange(data, 270)
  27. right_dist = getRange(data,90) # Ray perpendicular to right side of car
  28. a_right = getRange(data,(90 + theta)) # Ray at degree theta from right_dist
  29. a_left = getRange(data,(270 - theta))
  30. theta = math.radians(theta)
  31. # Calculating the deviation of steering(alpha) from right
  32. alpha_r = math.atan( (a_right * math.cos(theta) - right_dist)/ a_right * math.sin(theta) )
  33. curr_pos_r = right_dist * math.cos(alpha_r) # Present Position
  34. fut_pos_r = curr_pos_r + car_length * math.sin(alpha_r) # projection in Future Position
  35. # Calculating the deviation of steering(alpha) from left
  36. alpha_l = math.atan( (a_left * math.cos(theta) - left_dist)/ a_left * math.sin(theta) )
  37. curr_pos_l = left_dist * math.cos(alpha_l) # Present Position
  38. fut_pos_l = curr_pos_l + car_length * math.sin(alpha_l) # projection in Future Position
  39. error = - (fut_pos_r - fut_pos_l) # total error
  40. # print('error: ', error) #Testing
  41. # Sending PID error to Control
  42. msg = pid_input()
  43. msg.pid_error = error
  44. msg.pid_vel = dist_in_front # pid_vel used for distance in front
  45. pub.publish(msg)
  46. if __name__ == '__main__':
  47. print("Laser node started")
  48. rospy.init_node('dist_finder',anonymous = True)
  49. rospy.Subscriber("scan",LaserScan,callback)
  50. rospy.spin()

centre_wall_control.py 


  
  1. import rospy
  2. import math
  3. from race.msg import pid_input
  4. from ackermann_msgs.msg import AckermannDriveStamped
  5. kp = 0.27# 0.27 for porto
  6. kd = 0.0125#0.001 #0.09
  7. ki = 0 #0.5
  8. kp_vel = 0.9 #9 is using keep in middle
  9. kd_vel = 0.0001
  10. ki_error = 0
  11. prev_error = 0.0
  12. vel_input = 2.5 # base velocity
  13. angle = 0.0 # initial steering angle
  14. pub = rospy.Publisher("/drive", AckermannDriveStamped, queue_size=5)
  15. def control(data):
  16. global prev_error
  17. global vel_input
  18. global kp
  19. global kp_vel
  20. global kd
  21. global kd_vel
  22. global ki
  23. global angle
  24. dist_in_front = data.pid_vel
  25. front_obs = 1
  26. e = data.pid_error
  27. # Calculating deviation for lateral deviation from path
  28. kp_error = kp * e
  29. kd_error = kd * (e - prev_error)
  30. # Calculating error for velocity
  31. kp_vel_er = kp_vel * e
  32. kd_vel_er = kd * (e - prev_error)
  33. # ki_error = ki_error + (ki * e)
  34. vel_error = kp_vel_er + kd_vel_er
  35. pid_error = kp_error + kd_error
  36. min_angle=-20
  37. max_angle= 20
  38. # Heigher error results in lower velocity
  39. # while lower error results in heigher velocity
  40. if dist_in_front <= 5:
  41. front_obs = 3#abs(- ( (math.log10(dist_in_front/5)/math.log10(2)) +1 ))
  42. velo = vel_input + 1/ ( (abs(vel_error)*front_obs ))
  43. print("velo :", velo)
  44. #corrected steering angle
  45. angle = pid_error
  46. #print("raw velo:", velo) # Testing
  47. #Speed limit
  48. if velo > 50 :
  49. velo = 50
  50. # Filtering steering angle for Out-of-Range values
  51. if angle < min_angle:
  52. angle = min_angle
  53. elif angle > max_angle:
  54. angle = max_angle
  55. # print("filtered angle :" , angle) # Testing
  56. # Sending Drive information to Car
  57. msg = AckermannDriveStamped()
  58. msg.header.stamp = rospy.Time.now()
  59. msg.header.frame_id = ''
  60. msg.drive.speed = velo
  61. msg.drive.steering_angle = angle
  62. pub.publish(msg)
  63. if __name__ == '__main__':
  64. print("Starting control...")
  65. rospy.init_node('pid_controller', anonymous=True)
  66. rospy.Subscriber("error", pid_input, control)
  67. rospy.spin()

 (⊙﹏⊙)


如果按模板写不行吗???

参考1:


  
  1. #!/usr/bin/env python
  2. from __future__ import print_function
  3. import sys
  4. import math
  5. import numpy as np
  6. #ROS Imports
  7. import rospy
  8. from sensor_msgs.msg import Image, LaserScan
  9. from ackermann_msgs.msg import AckermannDriveStamped, AckermannDrive
  10. #PID CONTROL PARAMS
  11. kp = 1.0
  12. kd = 0.001
  13. ki = 0.005
  14. servo_offset = 0.0
  15. prev_error = 0.0
  16. error = 0.0
  17. integral = 0.0
  18. prev_time = 0.0
  19. #WALL FOLLOW PARAMS
  20. ANGLE_RANGE = 270 # Hokuyo 10LX has 270 degrees scan
  21. DESIRED_DISTANCE_RIGHT = 0.9 # meters
  22. DESIRED_DISTANCE_LEFT = 0.85
  23. VELOCITY = 1.5 # meters per second
  24. CAR_LENGTH = 1.0 # Traxxas Rally is 20 inches or 0.5 meters
  25. class WallFollow:
  26. """ Implement Wall Following on the car
  27. """
  28. def __init__(self):
  29. global prev_time
  30. #Topics & Subs, Pubs
  31. lidarscan_topic = '/scan'
  32. drive_topic = '/nav'
  33. prev_time = rospy.get_time()
  34. self.lidar_sub = rospy.Subscriber(lidarscan_topic, LaserScan, self.lidar_callback)
  35. self.drive_pub = rospy.Publisher(drive_topic, AckermannDriveStamped, queue_size = 10)
  36. def getRange(self, data, angle):
  37. # data: single message from topic /scan
  38. # angle: between -45 to 225 degrees, where 0 degrees is directly to the right
  39. # Outputs length in meters to object with angle in lidar scan field of view
  40. #make sure to take care of nans etc.
  41. #TODO: implement
  42. if angle >= -45 and angle <= 225:
  43. iterator = len(data) * (angle + 90) / 360
  44. if not np.isnan(data[int(iterator)]) and not np.isinf(data[int(iterator)]):
  45. return data[int(iterator)]
  46. def pid_control(self, error, velocity):
  47. global integral
  48. global prev_error
  49. global kp
  50. global ki
  51. global kd
  52. global prev_time
  53. angle = 0.0
  54. current_time = rospy.get_time()
  55. del_time = current_time - prev_time
  56. #TODO: Use kp, ki & kd to implement a PID controller for
  57. integral += prev_error * del_time
  58. angle = kp * error + ki * integral + kd * (error - prev_error) / del_time
  59. prev_error = error
  60. prev_time = current_time
  61. drive_msg = AckermannDriveStamped()
  62. drive_msg.header.stamp = rospy.Time.now()
  63. drive_msg.header.frame_id = "laser"
  64. drive_msg.drive.steering_angle = -angle
  65. if abs(angle) > math.radians(0) and abs(angle) <= math.radians(10):
  66. drive_msg.drive.speed = velocity
  67. elif abs(angle) > math.radians(10) and abs (angle) <= math.radians(20):
  68. drive_msg.drive.speed = 1.0
  69. else:
  70. drive_msg.drive.speed = 0.5
  71. self.drive_pub.publish(drive_msg)
  72. def followLeft(self, data, leftDist):
  73. #Follow left wall as per the algorithm
  74. #TODO:implement
  75. front_scan_angle = 125
  76. back_scan_angle = 180
  77. teta = math.radians(abs(front_scan_angle - back_scan_angle))
  78. front_scan_dist = self.getRange(data, front_scan_angle)
  79. back_scan_dist = self.getRange(data, back_scan_angle)
  80. alpha = math.atan2(front_scan_dist * math.cos(teta) - back_scan_dist, front_scan_dist * math.sin(teta))
  81. wall_dist = back_scan_dist * math.cos(alpha)
  82. ahead_wall_dist = wall_dist + CAR_LENGTH * math.sin(alpha)
  83. return leftDist - ahead_wall_dist
  84. def lidar_callback(self, data):
  85. """
  86. """
  87. error = self.followLeft(data.ranges, DESIRED_DISTANCE_LEFT) #TODO: replace with error returned by followLeft
  88. #send error to pid_control
  89. self.pid_control(error, VELOCITY)
  90. def main(args):
  91. rospy.init_node("WallFollow_node", anonymous=True)
  92. wf = WallFollow()
  93. rospy.sleep(0.1)
  94. rospy.spin()
  95. if __name__=='__main__':
  96. main(sys.argv)

参考2:


  
  1. #!/usr/bin/env python3
  2. import sys
  3. import math
  4. import numpy as np
  5. #ROS Imports
  6. import rospy
  7. from sensor_msgs.msg import LaserScan
  8. from ackermann_msgs.msg import AckermannDriveStamped
  9. #PID CONTROL PARAMS
  10. kp = 10
  11. kd = 70
  12. ki = 0.00001
  13. prev_error = 0
  14. integral = 0
  15. #WALL FOLLOW PARAMS
  16. ANGLE_RANGE = 270 # Hokuyo 10LX has 270 degrees scan
  17. class WallFollow:
  18. """ Implement Wall Following on the car
  19. """
  20. def __init__(self):
  21. self.rate = rospy.Rate(10)
  22. self.lidar_sub = rospy.Subscriber('/scan' , LaserScan, self.lidar_callback, queue_size = 1)
  23. self.drive_pub = rospy.Publisher('/nav', AckermannDriveStamped, queue_size = 1)
  24. def getRange(self, data, angle, distance):
  25. self.Dt_max = False
  26. angle_btwnScan = 70
  27. future_dist = 0.55
  28. theta = angle[int((180-angle_btwnScan+45)/ANGLE_RANGE * len(angle))]
  29. a = distance[int((180-angle_btwnScan+45)/ANGLE_RANGE * len(angle))]
  30. b = distance[int((180+45)/ANGLE_RANGE * len(angle))]
  31. alpha = math.atan((a*np.cos(np.pi-theta) - b)/a*np.sin(np.pi-theta))
  32. Dt = b*np.cos(alpha)
  33. Dt1 = Dt + future_dist*np.sin(alpha)
  34. # check condition for bottom part of map
  35. a2 = a = distance[int((180-15+45)/ANGLE_RANGE * len(angle))]
  36. theta2 = angle[int((180-15+45)/ANGLE_RANGE * len(angle))]
  37. alpha2 = math.atan((a2*np.cos(np.pi-theta2) - b)/a2*np.sin(np.pi-theta2))
  38. bot_error = a2*np.cos(np.pi-theta2+alpha2)
  39. Dt2 = b*np.cos(alpha2)
  40. self.bot_state = math.isclose(bot_error, Dt2, rel_tol = 0.1)
  41. if Dt > 1.1:
  42. self.Dt_max = True
  43. return Dt1
  44. def filter_scan(self, scan_msg):
  45. angle_range = enumerate(scan_msg.ranges)
  46. filter_data = [[count*scan_msg.angle_increment, val] for count, val in angle_range if not np.isinf(val) and not np.isnan(val)]
  47. filter_data = np.array(filter_data)
  48. angles = filter_data[:,0]
  49. distance = filter_data[:,1]
  50. filter_angle = np.array([])
  51. idx = 0
  52. start_idx = 0
  53. end_idx = 0
  54. for angle in angles:
  55. if (not 0 <= angle < np.pi/4) and (not 7*np.pi/4 < angle <= 2*np.pi):
  56. if start_idx == 0:
  57. start_idx = idx
  58. angle -= np.pi/2
  59. filter_angle = np.append(filter_angle, angle)
  60. if end_idx == 0 and angle > 7*np.pi/4:
  61. end_idx = idx - 1
  62. idx += 1
  63. distance = distance[start_idx: end_idx+1]
  64. return filter_angle, distance
  65. def pid_control(self, error):
  66. global integral
  67. global prev_error
  68. global kp
  69. global ki
  70. global kd
  71. integral += error
  72. angle = kp*error + kd*(error - prev_error) + ki*integral
  73. prev_error = error
  74. if self.bot_state == True and self.Dt_max == True:
  75. angle = -0.01*np.pi/180
  76. if -np.pi/18 < angle < np.pi/18:
  77. velocity = 1.5
  78. elif -np.pi/9 < angle <= -np.pi/18 or np.pi/18 <= angle < np.pi/9:
  79. velocity = 1
  80. else:
  81. velocity = 0.5
  82. drive_msg = AckermannDriveStamped()
  83. drive_msg.header.stamp = rospy.Time.now()
  84. drive_msg.header.frame_id = "laser"
  85. drive_msg.drive.steering_angle = angle
  86. drive_msg.drive.speed = velocity
  87. self.drive_pub.publish(drive_msg)
  88. def followLeft(self, leftDist):
  89. distToWall = 0.55
  90. error = -(distToWall - leftDist)
  91. return error
  92. def lidar_callback(self, data):
  93. try:
  94. angle, distance = self.filter_scan(data)
  95. Dt1 = self.getRange(data, angle, distance)
  96. error = self.followLeft(Dt1)
  97. self.pid_control(error)
  98. except rospy.ROSException as e:
  99. # rospy.logerr(e)
  100. pass
  101. def main(args):
  102. rospy.init_node("WallFollow_node", anonymous=True)
  103. wf = WallFollow()
  104. rospy.spin()
  105. if __name__=='__main__':
  106. main(sys.argv)

试一试看:


  
  1. # Keyboard characters for toggling mux
  2. joy_key_char: "j"
  3. keyboard_key_char: "k"
  4. brake_key_char: "b"
  5. random_walk_key_char: "r"
  6. nav_key_char: "n"
  7. # **Add button for new planning method here**
  8. new_key_char: "z"

 


 

文章来源: zhangrelay.blog.csdn.net,作者:zhangrelay,版权归原作者所有,如需转载,请联系作者。

原文链接:zhangrelay.blog.csdn.net/article/details/124547642

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。