B样条曲线简介及其python实现

1. B样条曲线的计算公式

B 样条曲线 \(P(u)\) 可以表示为控制点和基函数的加权和: \[ P(u)=\sum_{i=0}^{n} {p_i B_{i,k}(u)} \ ,u\in \left[ u_{k-1}, u_{k+1} \right] \tag{1} \]

2. 符号解释

2.1 \(P(u)\)

B 样条曲线上的点,也是我们所要求的结果。

2.2 \(p_i\)

控制点, \(i=0,1,2,..,n\),一共有 \(n+1\) 个。

2.3 \(B_{i,k}\)

基函数

2.4 \(u\)

相当于自变量,有效区间为 \(\left[ u_{k-1}, u_{k+1} \right]\)

2.5 \(k\)

B 样条曲线的次数,这里要注意次数和阶数的区别和联系:次数 = 阶数 - 1。

对于 B 样条的次数 \(k\),必须满足: \[k = m - n - 1 \tag{2}\] 其中 \(m\)节点 (knots) 将 B 样条曲线划分的段数,\(n\) 为控制点的个数减一。

上面说的节点就是划分 B 样条的比例,由节点组成的一组向量就成为节点矢量,例如 [0, 0.2, 0.4, 0.6, 0.8, 1]。不同的节点矢量进而产生了不同的 B 样条种类,例如均匀 B 样条、准均匀 B 样条、分段 B 样条以及非均匀 B 样条等等。

3. 基函数的计算

de Boor-Cox递归方法
\[ B_{i,k}(u)=\frac{u-u_i}{u_{i+1}-u_i}B_{i,k-1}(u)+\frac{u_{i+k}-u}{u_{i+k}-u_{i+1}}B_{i+1,k-1}(u) \tag{3} \]

这里需要给出说明,不带下标的 \(u\) 指的是 2.4 中的 \(u\),而带下标则指的是节点矢量。

在代码实现中,我们规定 \(\frac{0}{0}=0\)

4. 节点矢量 (knots) 的计算

节点矢量的取值可以是在 0 到 1 之间,也可以是其他范围,但是在代码是实现的时候一定要注意前后保持一致。
本文使用节点矢量取值为 0 到 1,这样也可以表示比例嘛。

这里需要注意节矢量的长度,也就是 \(m\),通过式 (2) 可知,\(m=k+n+1\)

4.1 均匀节点 (uniform node)

显然,节点的分布是均匀的,故从 0 到 1按照节点矢量的长度均匀划分即可。
(因为要将曲线划分为 m 段,当然需要 m+1个点)

1
2
3
4
def uniform_node(control_points, k=3):
num, _ = control_points.shape
n = num - 1
return np.linspace(0, 1, n + k + 2) # m = n + k + 1

均匀 B 样条曲线不一定过首尾控制点,并且在图像上是闭合的

4.2 准均匀节点 (quasi uniform node)

准均匀节点可以目的在于对曲线的端点进行行为控制,通过设计节点矢量,使得生成的 B 样条曲线经过首尾控制点。
\(k\) 次准均匀节点矢量中,两端节点具有重复度 \(k+1\),所有内节点呈现均匀分布。
在代码实现中,我们可以让节点矢量首尾分别为 \(k\) 个 0 和 1,然后中间 \(n-k+2\) 为 0 到 1 的均匀分布就行了,即 \[ \nonumber \left[ \begin{matrix} \underset{k}{\underbrace{0,0,...,0}},\ \underset{n-k+2}{\underbrace{0,...,1}},\ \underset{k}{\underbrace{1,1,...,1}}\\ \end{matrix} \right] \]

python
1
2
3
4
5
def quasi_uniform_node(control_points, k=3):
num, _ = control_points.shape
n = num - 1
mid = np.linspace(0, 1, n - k + 2)
return np.r_[np.zeros(k), mid, np.ones(k)] # 拼接

4.3 分段节点 (piecewise node)

基于该节点矢量的 B 样样条曲线又称为分段 Bezier 曲线,是一组顺序首尾相接且同为 \(k\) 次的 Bezier 曲线。
\(k\) 次的分段节点矢量中,首末端节点重复度依旧为 \(k+1\),内节点重复度为 \(k\)

1
2
3
4
5
6
7
8
9
10
def piecewise_node(control_points, k=3):
num, _ = control_points.shape
n = num - 1

## restricted condition
assert (n - k) % k == 0, "input is valid."

tmp = np.linspace(0, 1, int((n - k) / k + 2))
mid = np.r_[tmp[0], np.repeat(tmp[1:-1], k), tmp[-1]]
return np.r_[np.zeros(k), mid, np.ones(k)]

需要注意 \(n-k\) 必须是 \(k\) 的整数倍,否则不能生成曲线。

4.4 非均匀节点 (non-uniform node)

Hartley-Judd 算法 首尾重合度为 \(k+1\),内节点定义为:

\[ \begin{cases} t_k=0\\ t_i=\sum_{j=k+1}^i{\bigl( t_j-t_{j-1} \bigr)} \\ t_{n+1}=1\\ \end{cases} \tag{4} \] \[ t_i-t_{i-1}=\frac{\sum_{j=i-k}^{i-1}{l_j}}{\sum_{i=k+1}^{n+1}{\sum_{j=i-k}^{i-1}{l_j}}} \tag{5} \]

python
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
def non_uniform_node(control_points, k=3):
num, _ = control_points.shape
n = num - 1

l = np.sqrt(np.sum(np.diff(control_points, axis=0) ** 2, axis=1))
ll = l[0 : len(l) - 1] + l[1::]
L = np.sum(ll)

mid_size = n - k
mid = np.zeros(mid_size)

for i in range(mid_size):
mid[i] = np.sum(ll[0 : i + 1]) / L

knots = np.r_[np.zeros(k + 1), mid, np.ones(k + 1)]

return knots

5. python 实现

python
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
import numpy as np

def uniform_node(control_points, k=3):
num, _ = control_points.shape
n = num - 1

return np.linspace(0, 1, n + k + 2)


def quasi_uniform_node(control_points, k=3):
num, _ = control_points.shape
n = num - 1
mid = np.linspace(0, 1, n - k + 2)

return np.r_[np.zeros(k), mid, np.ones(k)]


def piecewise_node(control_points, k=3):
num, _ = control_points.shape
n = num - 1

## restricted condition
assert (n - k) % k == 0, "input is valid."

tmp = np.linspace(0, 1, int((n - k) / k + 2))
mid = np.r_[tmp[0], np.repeat(tmp[1:-1], k), tmp[-1]]

return np.r_[np.zeros(k), mid, np.ones(k)]


def non_uniform_node(control_points, k=3):
"""
Hartley-Judd algorithem
"""
num, _ = control_points.shape
n = num - 1

l = np.sqrt(np.sum(np.diff(control_points, axis=0) ** 2, axis=1))
ll = l[0 : len(l) - 1] + l[1::]
L = np.sum(ll)

mid_size = n - k
mid = np.zeros(mid_size)

for i in range(mid_size):
mid[i] = np.sum(ll[0 : i + 1]) / L

knots = np.r_[np.zeros(k + 1), mid, np.ones(k + 1)]

return knots


def cal_B(i, k, knots, u):
"""
de Boor-Cox recursion
Args:
i (int): ith point idx
k (int): degree of b-spline , equal to ord - 1
knots (ndarray): 1 dim
u : independent variable

Returns:
B: B_{i,k}
"""
if k == 1:
B = 1 if knots[i] <= u <= knots[i + 1] else 0 ##
else:
coef1 = coef2 = 0
if knots[i + k - 1] - knots[i] != 0:
coef1 = (u - knots[i]) / (knots[i + k - 1] - knots[i])
if knots[i + k] - knots[i + 1] != 0:
coef2 = (knots[i + k] - u) / (knots[i + k] - knots[i + 1])

B = coef1 * cal_B(i, k - 1, knots, u) + coef2 * cal_B(i + 1, k - 1, knots, u)

return B


def cal_curve(control_points, knots, t):

num, dims = control_points.shape
n = num - 1
k = len(knots) - n - 1 # degree of b-spline

N = len(t)
P = np.zeros((N, dims))

for idx in range(N):
u = t[idx]
for i in range(0, num):
P[idx, :] += control_points[i, :] * cal_B(i, k, knots, u)
return P


if __name__ == "__main__":

# 体验 https://superjerryshen.github.io/b-spline-demos/##/uniform-b-spline-of-order-3

import matplotlib.pyplot as plt

control_points = np.array([[50, 50], [100, 300], [300, 100], [380, 200], [400, 600], [500, 400], [300, 600]])

N = 500
t = np.linspace(0.0, 1.0, N)

uniform_knots = uniform_node(control_points)
quasi_knots = quasi_uniform_node(control_points)
piecewise_knots = piecewise_node(control_points)
non_knots = non_uniform_node(control_points)

P_uniform = cal_curve(control_points, uniform_knots, t)
P_quasi = cal_curve(control_points, quasi_knots, t)
P_piecewise = cal_curve(control_points, piecewise_knots, t)
P_non = cal_curve(control_points, non_knots, t)

fig, axs = plt.subplots(2, 2)

## 均匀 会闭合
axs[0, 0].scatter(control_points[:, 0], control_points[:, 1], color="C0", facecolors="none", label="control points")
axs[0, 0].plot(control_points[:, 0], control_points[:, 1], color="C0", linewidth=0.2)
axs[0, 0].plot(P_uniform[:, 0], P_uniform[:, 1], color="C3", label="uniform")
axs[0, 0].legend()

## 准均匀
axs[0, 1].scatter(control_points[:, 0], control_points[:, 1], color="C0", facecolors="none", label="control points")
axs[0, 1].plot(control_points[:, 0], control_points[:, 1], color="C0", linewidth=0.2)
axs[0, 1].plot(P_quasi[:, 0], P_quasi[:, 1], color="C3", label="quasi")
axs[0, 1].legend()

## 分段
axs[1, 0].scatter(control_points[:, 0], control_points[:, 1], color="C0", facecolors="none", label="control points")
axs[1, 0].plot(control_points[:, 0], control_points[:, 1], color="C0", linewidth=0.2)
axs[1, 0].plot(P_piecewise[:, 0], P_piecewise[:, 1], color="C3", label="piecewise")
axs[1, 0].legend()

## HJ 非均匀
axs[1,1].scatter(control_points[:, 0], control_points[:, 1], color="C0", facecolors="none", label="control points")
axs[1,1].plot(control_points[:, 0], control_points[:, 1], color="C0", linewidth=0.2)
axs[1,1].plot(P_non[:, 0], P_non[:, 1], color="C3", label="non uniform")
axs[1,1].legend()
plt.show()

效果:
BSpline

todo:
很诡异的一点,在 cal_curve 函数中计算 \(k\) 的时候,满足 \(k=m-n-1\) 的时候,后三种样条曲线都不经过首尾控制点,这是不符合预期的,但是用 \(k = (m+1) - n - 1\) 的时候,却和正常预期的结果一样,不知道是哪里出问题了。
有空了再来填补空缺

6. 参考

[1] 计算机图形学-中国农大-赵明-B站 8.5.1~8.6.2
[2] 详解样条曲线-CSDN
[3] B样条曲线和Nurbs曲线 图文并茂的理解节点和节点区间-知乎
[4]《计算几何算法与实现》—— 孔令德

-------------本文结束 感谢您的阅读-------------