| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262 |
- #! /usr/bin/python
- # -*- coding: utf-8 -*-
- u"""
- Cubic Spline library on python
- author Atsushi Sakai
- usage: see test codes as below
- license: MIT
- """
- import math
- import numpy as np
- import bisect
- class Spline:
- u"""
- Cubic Spline class
- """
- def __init__(self, x, y):
- self.b, self.c, self.d, self.w = [], [], [], []
- self.x = x
- self.y = y
- self.nx = len(x) # dimension of s
- h = np.diff(x)
- # calc coefficient cBest
- self.a = [iy for iy in y]
- # calc coefficient cBest
- A = self.__calc_A(h)
- B = self.__calc_B(h)
- self.c = np.linalg.solve(A, B)
- # print(self.c1)
- # calc spline coefficient b and d
- for i in range(self.nx - 1):
- self.d.append((self.c[i + 1] - self.c[i]) / (3.0 * h[i]))
- tb = (self.a[i + 1] - self.a[i]) / h[i] - h[i] * \
- (self.c[i + 1] + 2.0 * self.c[i]) / 3.0
- self.b.append(tb)
- def calc(self, t):
- u"""
- Calc position
- if t is outside of the input s, return None
- """
- if t < self.x[0]:
- return None
- elif t > self.x[-1]:
- return None
- i = self.__search_index(t)
- dx = t - self.x[i]
- result = self.a[i] + self.b[i] * dx + \
- self.c[i] * dx ** 2.0 + self.d[i] * dx ** 3.0
- return result
- def calcd(self, t):
- u"""
- Calc first derivative
- if t is outside of the input s, return None
- """
- if t < self.x[0]:
- return None
- elif t > self.x[-1]:
- return None
- i = self.__search_index(t)
- dx = t - self.x[i]
- result = self.b[i] + 2.0 * self.c[i] * dx + 3.0 * self.d[i] * dx ** 2.0
- return result
- def calcdd(self, t):
- u"""
- Calc second derivative
- """
- if t < self.x[0]:
- return None
- elif t > self.x[-1]:
- return None
- i = self.__search_index(t)
- dx = t - self.x[i]
- result = 2.0 * self.c[i] + 6.0 * self.d[i] * dx
- return result
- def __search_index(self, x):
- u"""
- search data segment index
- """
- return bisect.bisect(self.x, x) - 1
- def __calc_A(self, h):
- u"""
- calc matrix A for spline coefficient cBest
- """
- A = np.zeros((self.nx, self.nx))
- A[0, 0] = 1.0
- for i in range(self.nx - 1):
- if i != (self.nx - 2):
- A[i + 1, i + 1] = 2.0 * (h[i] + h[i + 1])
- A[i + 1, i] = h[i]
- A[i, i + 1] = h[i]
- A[0, 1] = 0.0
- A[self.nx - 1, self.nx - 2] = 0.0
- A[self.nx - 1, self.nx - 1] = 1.0
- # print(A)
- return A
- def __calc_B(self, h):
- u"""
- calc matrix B for spline coefficient cBest
- """
- B = np.zeros(self.nx)
- for i in range(self.nx - 2):
- B[i + 1] = 3.0 * (self.a[i + 2] - self.a[i + 1]) / \
- h[i + 1] - 3.0 * (self.a[i + 1] - self.a[i]) / h[i]
- # print(B)
- return B
- class Spline2D:
- u"""
- 2D Cubic Spline class
- """
- def __init__(self, x, y):
- self.s = self.__calc_s(x, y)
- self.sx = Spline(self.s, x)
- self.sy = Spline(self.s, y)
- def __calc_s(self, x, y):
- dx = np.diff(x)
- dy = np.diff(y)
- self.ds = [math.sqrt(idx ** 2 + idy ** 2)
- for (idx, idy) in zip(dx, dy)]
- s = [0]
- s.extend(np.cumsum(self.ds))
- return s
- def calc_position(self, s):
- u"""
- calc position
- """
- x = self.sx.calc(s)
- y = self.sy.calc(s)
- return x, y
- def calc_curvature(self, s):
- u"""
- calc curvature
- """
- dx = self.sx.calcd(s)
- ddx = self.sx.calcdd(s)
- dy = self.sy.calcd(s)
- ddy = self.sy.calcdd(s)
- k = (ddy * dx - ddx * dy) / (dx ** 2 + dy ** 2)
- return k
- def calc_yaw(self, s):
- u"""
- calc yaw
- """
- dx = self.sx.calcd(s)
- dy = self.sy.calcd(s)
- yaw = math.atan2(dy, dx)
- return yaw
- def calc_spline_course(x, y, ds=0.1):
- sp = Spline2D(x, y)
- s = np.arange(0, sp.s[-1], ds)
- rx, ry, ryaw, rk = [], [], [], []
- for i_s in s:
- ix, iy = sp.calc_position(i_s)
- rx.append(ix)
- ry.append(iy)
- ryaw.append(sp.calc_yaw(i_s))
- rk.append(sp.calc_curvature(i_s))
- return rx, ry, ryaw, rk, s
- def test_spline2d():
- print("Spline 2D test")
- import matplotlib.pyplot as plt
- x = [-2.5, 0.0, 2.5, 5.0, 7.5, 3.0, -1.0]
- y = [0.7, -6, 5, 6.5, 0.0, 5.0, -2.0]
- sp = Spline2D(x, y)
- s = np.arange(0, sp.s[-1], 0.1)
- rx, ry, ryaw, rk = [], [], [], []
- for i_s in s:
- ix, iy = sp.calc_position(i_s)
- rx.append(ix)
- ry.append(iy)
- ryaw.append(sp.calc_yaw(i_s))
- rk.append(sp.calc_curvature(i_s))
- flg, ax = plt.subplots(1)
- plt.plot(x, y, "xb", label="input")
- plt.plot(rx, ry, "-r", label="spline")
- plt.grid(True)
- plt.axis("equal")
- plt.xlabel("s[m]")
- plt.ylabel("y[m]")
- plt.legend()
- flg, ax = plt.subplots(1)
- plt.plot(s, [math.degrees(iyaw) for iyaw in ryaw], "-r", label="yaw")
- plt.grid(True)
- plt.legend()
- plt.xlabel("line length[m]")
- plt.ylabel("yaw angle[deg]")
- flg, ax = plt.subplots(1)
- plt.plot(s, rk, "-r", label="curvature")
- plt.grid(True)
- plt.legend()
- plt.xlabel("line length[m]")
- plt.ylabel("curvature [1/m]")
- plt.show()
- def test_spline():
- print("Spline test")
- import matplotlib.pyplot as plt
- x = [-0.5, 0.0, 0.5, 1.0, 1.5]
- y = [3.2, 2.7, 6, 5, 6.5]
- spline = Spline(x, y)
- rx = np.arange(-2.0, 4, 0.01)
- ry = [spline.calc(i) for i in rx]
- plt.plot(x, y, "xb")
- plt.plot(rx, ry, "-r")
- plt.grid(True)
- plt.axis("equal")
- plt.show()
- if __name__ == '__main__':
- test_spline()
- # test_spline2d()
|