刚体动力学算法 Chapter 5 逆动力学

Chapter 5 Inverse Dynamics

逆动力学是指在刚体系统中寻找产生给定加速度所需的力。逆动力学计算在运动控制系统、轨迹设计和优化(例如用于机器人和动画角色)、机械设计以及某些正向动力学算法中都有应用。最后一个应用在第6章中介绍。逆动力学计算可以用以下方程表示:
τ = I D ( model ⁡ , q , q ˙ , q ¨ ) , \boldsymbol{\tau}=\mathrm{ID}(\operatorname{model}, \boldsymbol{q}, \dot{\boldsymbol{q}}, \ddot{\boldsymbol{q}}), τ=ID(model,q,q˙,q¨),

其中 q , q ˙ , q ¨ \boldsymbol{q}, \dot{\boldsymbol{q}}, \ddot{\boldsymbol{q}} q,q˙,q¨ τ \boldsymbol{\tau} τ 分别是广义位置、速度、加速度和力的向量,而 model 是特定刚体系统的系统模型。

本章介绍了一种计算运动树逆动力学的算法。闭环系统的算法将在第8章中出现。运动树的逆动力学计算特别容易,这就是我们为什么从这个问题开始的原因。该算法称为递归牛顿-欧拉算法,它是最简单、最高效的算法。它的计算复杂度为 O ( n ) O(n) O(n),这意味着计算量与树中的刚体或关节变量数量呈线性增长。首先介绍基本形式的算法,然后添加实现细节,最后将空间向量版本与原始的3D向量版本进行比较。本章还涵盖算法复杂度的主题,并解释了为什么递归算法如此高效。

5.1 算法复杂度

我们将算法的计算成本定义为其操作数量,即在执行算法过程中执行的算术操作次数。如果算法涉及实数运算(如动力学算法),则习惯上仅计算浮点操作。因此,忽略了整数操作,例如计算数组索引或递增循环变量所需的操作。

算法的计算复杂度是衡量成本如何随问题规模增长的一种指标。我们使用“大O”符号来指定这个数量。如果一个算法的复杂度被描述为 O ( n p ) O\left(n^{p}\right) O(np),其中 n n n 是问题的规模,那么当 n → ∞ n \rightarrow \infty n 时,成本将与 n p n^{p} np 成比例增长。更准确地说,如果存在正常数 C min ⁡ , C max ⁡ C_{\min },C_{\max } CminCmax n min ⁡ n_{\min } nmin,使得计算成本至少为 C min ⁡ f ( n ) C_{\min } f(n) Cminf(n),但对于所有 n > n min ⁡ n>n_{\min } n>nmin,不超过 C max ⁡ f ( n ) C_{\max } f(n) Cmaxf(n) ,则称算法具有 O ( f ( n ) ) O(f(n)) O(f(n)) 复杂度。

表达式 O ( f ( n ) ) O(f(n)) O(f(n)) 可以解释为所有在 n → ∞ n \rightarrow \infty n 时与 f ( n ) f(n) f(n) 成比例增长的函数集合。因此,像“算法 X X X O ( f ( n ) ) O(f(n)) O(f(n)) ”这样的说法意味着 X X X 的成本,表示为 n n n 的函数,是集合 O ( f ( n ) ) O(f(n)) O(f(n)) 的元素。类似地,表达式 O ( f ( n ) ) = O ( g ( n ) ) O(f(n))=O(g(n)) O(f(n))=O(g(n)) 是一个集合相等,意味着 g ( n ) ∈ O ( f ( n ) ) g(n) \in O(f(n)) g(n)O(f(n)) 且反之亦然。在引用复杂度时,我们选择集合中最简单的函数作为 O O O 的参数。例如,如果算法的精确成本由多项式 a 0 + a 1 n + ⋯ + a p n p a_{0}+a_{1} n+\cdots+a_{p} n^{p} a0+a1n++apnp 给出,则与该多项式成比例增长的最简单函数是 n p n^{p} np,因此我们引用复杂度为 O ( n p ) O\left(n^{p}\right) O(np) 。通常,只有 f f f 中增长最快的组成部分才重要。如果 f f f 收敛到常数,则称 f f f O ( 1 ) O(1) O(1)

算法的复杂度取决于问题规模的衡量方式。例如,矩阵求逆的复杂度被认为是 O ( n 3 ) O\left(n^{3}\right) O(n3) ,因为求逆一个 n × n n \times n n×n 矩阵的成本随着 n → ∞ n \rightarrow \infty n 而以 n 3 n^{3} n3 成比例增长。这个数字依赖于 n n n 是矩阵的维度。如果 n n n 是矩阵中元素的数量,那么复杂度数字将是 O ( n 1.5 ) O\left(n^{1.5}\right) O(n1.5)。更一般地,如果 m m m n n n 是问题规模的两种不同衡量方式,并且 m = g ( n ) m=g(n) m=g(n),那么 O ( f ( m ) ) = O ( f ( g ( n ) ) ) O(f(m))=O(f(g(n))) O(f(m))=O(f(g(n)))

对于动力学算法,问题的规模是刚体系统的规模。如果系统是一个运动链,那么有两种合理的系统规模衡量方式:刚体数量 N B N_{B} NB 和自由度数量 n n n。如果链中的每个关节至少有一个自由度,则这两个数字满足 N B ≤ n ≤ 6 N B N_{B} \leq n \leq 6 N_{B} NBn6NB,这意味着

O ( N B p ) = O ( n p ) . O\left(N_{B}^{p}\right)=O\left(n^{p}\right) . O(NBp)=O(np).

因此,在引用复杂度时,无需区分 N B N_{B} NB n n n

5.2 递归关系

现代动力学算法是递归的,它们比非递归的前身快几个数量级。它们之所以被称为递归,是因为它们利用递归关系来计算中间结果的序列。正是递归关系的使用导致了它们的高效性;因此,让我们花一点时间来看看这些递归关系是如何工作的。递归关系是一个公式,它根据前面的元素定义了序列中的下一个元素。这个公式连同一组初始值提供了序列的递归定义。例如,斐波那契数列 0 , 1 , 1 , 2 , 3 , 5 , 8 , 13 , … 0,1,1,2,3,5,8,13, \ldots 0,1,1,2,3,5,8,13,可以通过递归关系 F i = F i − 1 + F i − 2 F_{i}=F_{i-1}+F_{i-2} Fi=Fi1+Fi2 和初始值 F 0 = 0 F_{0}=0 F0=0 F 1 = 1 F_{1}=1 F1=1 来定义。对于刚体系统,在根据 第4.1.2节 的规则编号的情况下,刚体速度序列 v 0 , v 1 , v 2 , … \boldsymbol{v}_{0}, \boldsymbol{v}_{1}, \boldsymbol{v}_{2}, \ldots v0,v1,v2,可以通过递归关系 v i = v λ ( i ) + S i q ˙ i \boldsymbol{v}_{i}=\boldsymbol{v}_{\lambda(i)}+\boldsymbol{S}_{i} \dot{\boldsymbol{q}}_{i} vi=vλ(i)+Siq˙i 和给定的 v 0 \boldsymbol{v}_{0} v0 的初始值来定义。

让我们来看看使用递归关系和不使用递归关系计算刚体速度的任务。为了简化问题,假设系统是一个无分支的运动链,具有1自由度关节。这意味着 λ ( i ) = i − 1 \lambda(i)=i-1 λ(i)=i1 ,并且关节 i i i 上的速度是关节轴向量 s i s_{i} si 与标量关节速度变量 q ˙ i \dot{q}_{i} q˙i 的乘积。刚体速度的递归关系如下:

v i = v i − 1 + s i q ˙ i (5.2) \boldsymbol{v}_{i}=\boldsymbol{v}_{i-1}+\boldsymbol{s}_{i} \dot{q}_{i} \tag{5.2} vi=vi1+siq˙i(5.2)

在初始条件 v 0 = 0 \boldsymbol{v}_{0}=\mathbf{0} v0=0 下,刚体速度的非递归公式为:

v i = ∑ j = 1 i s j q ˙ j (5.3) \boldsymbol{v}_{i}=\sum_{j=1}^{i} \boldsymbol{s}_{j} \dot{q}_{j} \tag{5.3} vi=j=1isjq˙j(5.3)

m \mathrm{m} m a \mathrm{a} a 分别表示标量-向量乘法和向量加法的成本。方程5.2 的一次应用成本为 m + a \mathrm{m}+\mathrm{a} m+a,方程5.3 的成本为 i   m + ( i − 1 ) i \mathrm{~m}+(i-1) i m+(i1) a。通过方程5.2 计算前 n n n 个刚体速度的成本为 n (   m + a ) n(\mathrm{~m}+\mathrm{a}) n( m+a),通过方程5.3 的成本为 1 2 ( n ( n + 1 ) m + n ( n − 1 ) a ) \frac{1}{2}(n(n+1) \mathrm{m}+n(n-1) \mathrm{a}) 21(n(n+1)m+n(n1)a)。因此,使用方程5.2 可以得到高效的 O ( n ) O(n) O(n) 计算,而方程5.3 导致低效的 O ( n 2 ) O\left(n^{2}\right) O(n2) 计算。

这种低效的原因并不难理解。如果我们使用方程5.3 进行计算,那么我们最终需要计算的是
v 1 = s 1 q ˙ 1 v 2 = s 1 q ˙ 1 + s 2 q ˙ 2 ⋮ ⋮ v n = s 1 q ˙ 1 + s 2 q ˙ 2 + ⋯ + s n q ˙ n . \begin{aligned} \boldsymbol{v}_{1} & =\boldsymbol{s}_{1} \dot{q}_{1} \\ \boldsymbol{v}_{2} & =\boldsymbol{s}_{1} \dot{q}_{1}+\boldsymbol{s}_{2} \dot{q}_{2} \\ & \vdots \\ & \vdots \\ \boldsymbol{v}_{n} & =\boldsymbol{s}_{1} \dot{q}_{1}+\boldsymbol{s}_{2} \dot{q}_{2}+\cdots+\boldsymbol{s}_{n} \dot{q}_{n} . \end{aligned} v1v2vn=s1q˙1=s1q˙1+s2q˙2=s1q˙1+s2q˙2++snq˙n.

在这种情况下, s 1 q ˙ 1 \boldsymbol{s}_{1} \dot{q}_{1} s1q˙1 的乘积被计算了 n n n 次,而实际上只需要计算一次。同样地, s 2 q ˙ 2 s_{2} \dot{q}_{2} s2q˙2 的乘积被计算了 n − 1 n-1 n1 次,和 s 1 q ˙ 1 + s 2 q ˙ 2 \boldsymbol{s}_{1} \dot{q}_{1}+\boldsymbol{s}_{2} \dot{q}_{2} s1q˙1+s2q˙2 的和被计算了 n − 1 n-1 n1 次,依此类推。因此,递归关系提供了一种系统的方法来避免在计算相关量的序列时不必要的重复,这是其高效性的秘诀。

当计算更加复杂时,递归关系的计算优势甚至更大。例如,在无分支链中刚体加速度的递归关系是
a i = a i − 1 + s i q ¨ i + v i × s i q ˙ i (5.4) \boldsymbol{a}_{i}=\boldsymbol{a}_{i-1}+\boldsymbol{s}_{i} \ddot{q}_{i}+\boldsymbol{v}_{i} \times \boldsymbol{s}_{i} \dot{q}_{i} \tag{5.4} ai=ai1+siq¨i+vi×siq˙i(5.4)

这正是 公式5.2 的时间导数,而非递归公式则为:

a i = ∑ j = 1 i s j q ¨ j + ∑ j = 1 i ∑ k = 1 j − 1 ( s k q ˙ k ) × ( s j q ˙ j ) (5.5) \boldsymbol{a}_{i}=\sum_{j=1}^{i} \boldsymbol{s}_{j} \ddot{q}_{j}+\sum_{j=1}^{i} \sum_{k=1}^{j-1}\left(\boldsymbol{s}_{k} \dot{q}_{k}\right) \times\left(\boldsymbol{s}_{j} \dot{q}_{j}\right) \tag{5.5} ai=j=1isjq¨j+j=1ik=1j1(skq˙k)×(sjq˙j)(5.5)

(在这两种情况下,我们假设 s ˙ i = v i × s i \dot{\boldsymbol{s}}_{i}=\boldsymbol{v}_{i} \times \boldsymbol{s}_{i} s˙i=vi×si 。)通过公式5.4计算加速度是一个 O ( n ) O(n) O(n)的计算;但是通过公式5.5计算是一个 O ( n 3 ) O\left(n^{3}\right) O(n3)的计算。

公式5.3和5.5直接用其基本要素——关节轴向量、关节速度和加速度变量来表示身体速度和加速度。这样的方程被称为闭式形式方程。刚体系统的运动方程在闭式形式下可以表示为:

τ i = ∑ j = 1 n H i j q ¨ i + ∑ j = 1 n ∑ k = 1 n C i j k q ˙ j q ˙ k + g i (5.6) \tau_{i}=\sum_{j=1}^{n} H_{i j} \ddot{q}_{i}+\sum_{j=1}^{n} \sum_{k=1}^{n} C_{i j k} \dot{q}_{j} \dot{q}_{k}+g_{i} \tag{5.6} τi=j=1nHijq¨i+j=1nk=1nCijkq˙jq˙k+gi(5.6)

其中 H i j H_{i j} Hij是广义惯量矩阵的元素, C i j k C_{i j k} Cijk是离心力和科氏力的系数, g i g_{i} gi是重力项。早期的非递归算法通过类似这样的方程计算动力学,并且它们的计算复杂性通常为 O ( n 4 ) O\left(n^{4}\right) O(n4)。递归算法效率更高,其计算复杂性可以降低至 O ( n ) O(n) O(n)。当递归的牛顿-欧拉算法首次出现时,相对于非递归的前身——Uicker/Kahn算法,对于一个6自由度的机械臂,其速度约快100倍(Hollerbach, 1980)。在高效的递归算法发明之前,闭式形式方程是表示和评估运动方程的标准方式。

5.3 递归牛顿-欧拉算法

一般运动树的逆动力学可以按照以下三个步骤计算:

  1. 计算运动树中每个刚体的速度和加速度。
  2. 计算产生这些加速度所需的力。
  3. 计算从作用于刚体上的力传递到关节的力。

这些是递归牛顿-欧拉算法的步骤。现在让我们推导出每个步骤的方程,假设树由量 λ ( i ) , μ ( i ) , S i \lambda(i), \mu(i), \boldsymbol{S}_{i} λ(i),μ(i),Si I i \boldsymbol{I}_{i} Ii来建模,如第4章所解释的那样。

图5.1:作用在刚体 i 上的力

步骤1:设 v i \boldsymbol{v}_{i} vi a i \boldsymbol{a}_{i} ai 分别为刚体 i i i 的速度和加速度。在一个运动树中,任何刚体的速度可以递归地定义为其父刚体的速度和连接它与父刚体的关节处的速度之和。因此,
v i = v λ ( i ) + S i q ˙ i ⋅ ( v 0 = 0 ) (5.7) \boldsymbol{v}_{i}=\boldsymbol{v}_{\lambda(i)}+\boldsymbol{S}_{i} \dot{\boldsymbol{q}}_{i} \cdot \quad\left(\boldsymbol{v}_{0}=\mathbf{0}\right) \tag{5.7} vi=vλ(i)+Siq˙i(v0=0)(5.7)

(初始值以括号形式给出)。通过对该方程进行求导得到加速度的递推关系:

a i = a λ ( i ) + S i q ¨ i + S ˙ i q ˙ i . ( a 0 = 0 ) (5.8) \boldsymbol{a}_{i}=\boldsymbol{a}_{\lambda(i)}+\boldsymbol{S}_{i} \ddot{\boldsymbol{q}}_{i}+\dot{\boldsymbol{S}}_{i} \dot{\boldsymbol{q}}_{i} . \quad\left(\boldsymbol{a}_{0}=\mathbf{0}\right) \tag{5.8} ai=aλ(i)+Siq¨i+S˙iq˙i.(a0=0)(5.8)

使用这两个公式进行连续迭代,其中 i i i 的取值范围从 1 到 N B N_{B} NB,可以计算出运动树中每个刚体的速度和加速度。

步骤2:设 f i B \boldsymbol{f}_{i}^{B} fiB 为作用在刚体 i i i 上的净力。这个力与刚体 i i i 的加速度之间存在如下运动方程的关系:
f i B = I i a i + v i × ∗ I i v i (5.9) \boldsymbol{f}_{i}^{B}=\boldsymbol{I}_{i} \boldsymbol{a}_{i}+\boldsymbol{v}_{i} \times^{*} \boldsymbol{I}_{i} \boldsymbol{v}_{i} \tag{5.9} fiB=Iiai+vi×Iivi(5.9)

步骤3:参考图5.1,设 f i \boldsymbol{f}_{i} fi为从刚体 λ ( i ) \lambda(i) λ(i)传递到刚体 i i i的关节 i i i处的力, f i x \boldsymbol{f}_{i}^{x} fix为作用在刚体 i i i上的外力(如果有的话)。外力可以模拟各种环境影响,如力场、物理接触等。它们被视为算法的输入;也就是说,它们的值被假定为已知。刚体 i i i上的净力则为:
f i B = f i + f i x − ∑ j ∈ μ ( i ) f j , \boldsymbol{f}_{i}^{B}=\boldsymbol{f}_{i}+\boldsymbol{f}_{i}^{x}-\sum_{j \in \mu(i)} \boldsymbol{f}_{j}, fiB=fi+fixjμ(i)fj,

这个方程可以重新排列为关节力的递推关系:

f i = f i B − f i x + ∑ j ∈ μ ( i ) f j (5.10) \boldsymbol{f}_{i}=\boldsymbol{f}_{i}^{B}-\boldsymbol{f}_{i}^{x}+\sum_{j \in \mu(i)} \boldsymbol{f}_{j} \tag{5.10} fi=fiBfix+jμ(i)fj(5.10)

使用这个公式进行连续迭代,其中 i i i 的取值范围从 N B N_{B} NB递减至1,可以计算出运动树中每个关节处的空间力。对于这个公式,不需要初始值,因为任何没有子刚体的刚体都会使 μ ( i ) \mu(i) μ(i)为空集。

在计算了每个关节处的空间力之后,只需计算关节处的广义力,即:

τ i = S i T f i (5.11) \boldsymbol{\tau}_{i}=\boldsymbol{S}_{i}^{\mathrm{T}} \boldsymbol{f}_{i} \tag{5.11} τi=SiTfi(5.11)

(参考方程3.37)。方程5.7到5.11描述了递归牛顿-欧拉算法的最简形式。

外力可以用来模拟各种环境影响,包括重力。然而,可以更高效地将均匀重力场建模为基座的虚拟加速度。为此,我们只需将方程5.8的初始值替换为 a 0 = − a g \boldsymbol{a}_{0}=-\boldsymbol{a}_{g} a0=ag,其中 a g \boldsymbol{a}_{g} ag为重力加速度矢量。如果使用这个技巧,那么计算得到的 a i \boldsymbol{a}_{i} ai f i B \boldsymbol{f}_{i}^{B} fiB不再是刚体 i i i的真实加速度和净力,而是分别被重力加速度和重力力矢量所偏移。(刚体 i i i上的重力力为 I i a g \boldsymbol{I}_{i} \boldsymbol{a}_{g} Iiag)。

算法特点

方程5.7和5.8涉及从根部到叶子的数据传播;方程5.10涉及从叶子到根部的数据传播;而方程5.9和5.11描述了每个刚体局部的计算。因此,递归牛顿-欧拉算法是一个两遍过程:首先进行外部传递,计算速度和加速度;然后进行内部传递,计算关节力。计算刚体力可以分配给第一次或第二次传递中的任意一次。通常选择第一次传递。

这个算法不仅计算 τ i \boldsymbol{\tau}_{i} τi,还可以计算其他量,比如 v i \boldsymbol{v}_{i} vi f i \boldsymbol{f}_{i} fi,有时候也是有用的。例如,机械设计师的任务之一是选择足够强度的关节轴承以承受运行期间的动态反作用力。这些力可以从 f i \boldsymbol{f}_{i} fi中计算得到。

通过将某些输入设为零,可以计算出动力学的特定分量。例如,如果将外部力、关节速度和加速度项设为零,但保留虚拟基底加速度,则算法只计算重力补偿项——静态平衡重力的项。这样的项在机器人控制系统中被使用。另外,如果将重力、加速度和外部力项都设为零,但保留关节速度,那么算法只计算科氏力和离心力的项。如果对效率有要求,那么可以使用简化版本的算法来执行这些专门的计算。例如,如果目标是仅计算重力补偿项,那么算法简化为:

f i = − I i a g + ∑ j ∈ μ ( i ) f j τ i = S i T f i . \begin{aligned} \boldsymbol{f}_{i} & =-\boldsymbol{I}_{i} \boldsymbol{a}_{g}+\sum_{j \in \mu(i)} \boldsymbol{f}_{j} \\ \boldsymbol{\tau}_{i} & =\boldsymbol{S}_{i}^{\mathrm{T}} \boldsymbol{f}_{i} . \end{aligned} fiτi=Iiag+jμ(i)fj=SiTfi.

要计算实际的重力项,将 − I i a g -\boldsymbol{I}_{i} \boldsymbol{a}_{g} Iiag替换为 I i a g \boldsymbol{I}_{i} \boldsymbol{a}_{g} Iiag

刚体坐标下的方程

方程5.7到5.11表达了算法而不涉及任何特定的坐标系。在实际应用中,如果将与刚体 i i i相关的计算以刚体 i i i的坐标系(如第4.2节中定义的)进行,则算法效果最佳。这是算法的刚体坐标(或链路坐标)版本。

让我们将递归牛顿-欧拉算法的方程转换为刚体坐标。量 v i , a i , S i , I i , f i \boldsymbol{v}_{i}, \boldsymbol{a}_{i}, \boldsymbol{S}_{i}, \boldsymbol{I}_{i}, \boldsymbol{f}_{i} vi,ai,Si,Ii,fi f i B \boldsymbol{f}_{i}^{B} fiB现在都被认为是以刚体 i i i的坐标系表示的。注意,系统模型已经以这些坐标提供了 S i \boldsymbol{S}_{i} Si I i \boldsymbol{I}_{i} Ii。用刚体坐标表示,方程5.7和5.8变为:

v i = i X λ ( i ) v λ ( i ) + S i q ˙ i ( v 0 = 0 ) \boldsymbol{v}_{i}={ }^{i} \boldsymbol{X}_{\lambda(i)} \boldsymbol{v}_{\lambda(i)}+\boldsymbol{S}_{i} \dot{\boldsymbol{q}}_{i} \quad\left(\boldsymbol{v}_{0}=\mathbf{0}\right) vi=iXλ(i)vλ(i)+Siq˙i(v0=0)

a i = i X λ ( i ) a λ ( i ) + S i q ¨ i + S ∘ i q ˙ i + v i × S i q ˙ i . ( a 0 = − a g ) \boldsymbol{a}_{i}={ }^{i} \boldsymbol{X}_{\lambda(i)} \boldsymbol{a}_{\lambda(i)}+\boldsymbol{S}_{i} \ddot{\boldsymbol{q}}_{i}+\stackrel{\circ}{\boldsymbol{S}}_{i} \dot{\boldsymbol{q}}_{i}+\boldsymbol{v}_{i} \times \boldsymbol{S}_{i} \dot{\boldsymbol{q}}_{i} . \quad\left(\boldsymbol{a}_{0}=-\boldsymbol{a}_{g}\right) ai=iXλ(i)aλ(i)+Siq¨i+Siq˙i+vi×Siq˙i.(a0=ag)

与原方程相比,进行了以下更改:插入了坐标转换,将 v λ ( i ) \boldsymbol{v}_{\lambda(i)} vλ(i) a λ ( i ) \boldsymbol{a}_{\lambda(i)} aλ(i)从刚体 λ ( i ) \lambda(i) λ(i)的坐标系转换为刚体 i i i的坐标系;并且将矩阵 S ˙ i \dot{\boldsymbol{S}}_{i} S˙i替换为 S ∘ i + v i × S i \stackrel{\circ}{\boldsymbol{S}}_{i}+\boldsymbol{v}_{i} \times \boldsymbol{S}_{i} Si+vi×Si,其中 S ∘ i \stackrel{\circ}{\boldsymbol{S}}_{i} Si是在刚体坐标中的 S i \boldsymbol{S}_{i} Si的显式导数。我们还利用这个机会,将原始的初始加速度值替换为 a 0 = − a g \boldsymbol{a}_{0}=-\boldsymbol{a}_{g} a0=ag以模拟重力的影响。

关节计算例程jcalc能够以刚体 i i i的坐标系提供量 S i \boldsymbol{S}_i Si v J i = S i q ˙ i \boldsymbol{v}_{\mathrm{J} i}=\boldsymbol{S}_i \dot{\boldsymbol{q}}_i vJi=Siq˙i c J i = S ˙ i q ˙ i \boldsymbol{c}_{\mathrm{J} i}=\dot{\boldsymbol{S}}_i \dot{\boldsymbol{q}}_i cJi=S˙iq˙i。因此,为了利用可用的数据,我们修改方程5.14和5.15如下:

v J i = S i q ˙ i c J i = S ∘ i q ˙ i v i = i X λ ( i ) v λ ( i ) + v J i ( v 0 = 0 ) a i = i X λ ( i ) a λ ( i ) + S i q ¨ i + c J i + v i × v J i . ( a 0 = − a g ) (5.16-19) \begin{aligned} \boldsymbol{v}_{\mathrm{J} i} & =\boldsymbol{S}_{i} \dot{\boldsymbol{q}}_{i} \\ \boldsymbol{c}_{\mathrm{J} i} & =\stackrel{\circ}{\boldsymbol{S}}_{i} \dot{\boldsymbol{q}}_{i} \\ \boldsymbol{v}_{i} & ={ }^{i} \boldsymbol{X}_{\lambda(i)} \boldsymbol{v}_{\lambda(i)}+\boldsymbol{v}_{\mathrm{J} i} \quad\left(\boldsymbol{v}_{0}=\mathbf{0}\right) \\ \boldsymbol{a}_{i} & ={ }^{i} \boldsymbol{X}_{\lambda(i)} \boldsymbol{a}_{\lambda(i)}+\boldsymbol{S}_{i} \ddot{\boldsymbol{q}}_{i}+\boldsymbol{c}_{\mathrm{J} i}+\boldsymbol{v}_{i} \times \boldsymbol{v}_{\mathrm{J} i} . \quad\left(\boldsymbol{a}_{0}=-\boldsymbol{a}_{g}\right) \end{aligned} \tag{5.16-19} vJicJiviai=Siq˙i=Siq˙i=iXλ(i)vλ(i)+vJi(v0=0)=iXλ(i)aλ(i)+Siq¨i+cJi+vi×vJi.(a0=ag)(5.16-19)

在刚体坐标系版本中,方程5.9和5.11保持不变,但是方程5.10变为:

f i = f i B − i X 0 ∗ f i x + ∑ j ∈ μ ( i ) i X j ∗ f j (5.20) \boldsymbol{f}_{i}=\boldsymbol{f}_{i}^{B}-{ }^{i} \boldsymbol{X}_{0}^{*} \boldsymbol{f}_{i}^{x}+\sum_{j \in \mu(i)}{ }^{i} \boldsymbol{X}_{j}^{*} \boldsymbol{f}_{j} \tag{5.20} fi=fiBiX0fix+jμ(i)iXjfj(5.20)

在这里插入图片描述

在这个方程中,我们假设外部力来自系统外部,并且假设它们以绝对坐标(即刚体0的坐标)表示。方程5.16到5.20与方程5.9和5.11一起,是递归牛顿-欧拉算法的刚体坐标版本的方程。

算法细节

表格5.1展示了算法的刚体坐标版本的伪代码,同时给出了基本版本和刚体坐标版本的方程。请注意,刚体坐标版本中的符号 v i , a i \boldsymbol{v}_{i}, \boldsymbol{a}_{i} vi,ai 等是以刚体坐标表示的,而基本版本中相同的符号表示抽象向量或未指定公共坐标系中的坐标向量。

从伪代码中的两个for循环可以看出算法的两次通过结构。符号 X J , v J \boldsymbol{X}_{\mathrm{J}}, \boldsymbol{v}_{\mathrm{J}} XJ,vJ c J \boldsymbol{c}_{\mathrm{J}} cJ 是局部变量,每次第一个循环迭代时都会取新值。如果树中包含速度变量不是位置变量的导数的关节,则只需将符号 q ˙ i \dot{\boldsymbol{q}}_{i} q˙i q ¨ i \ddot{\boldsymbol{q}}_{i} q¨i 分别替换为 α i \boldsymbol{\alpha}_{i} αi α ˙ i \dot{\boldsymbol{\alpha}}_{i} α˙i。同样地,如果树中包含任何显式依赖于时间的关节,则 jcalc 将使用方程3.32和3.41而不是5.16和5.17自动计算 v J \boldsymbol{v}_{\mathrm{J}} vJ c J \boldsymbol{c}_{\mathrm{J}} cJ;伪代码中唯一的变化是将当前时间作为jcalc的第四个参数传递。 i X 0 { }^{i} \boldsymbol{X}_{0} iX0 仅用于将外部力从基座标转换为刚体坐标。如果没有外部力,则可以省略计算这些变换的if语句。

在实现方程5.20时,我们将其从使用 μ ( i ) \mu(i) μ(i) 的计算转换为使用 λ ( i ) \lambda(i) λ(i) 的计算。方程5.20的直译伪代码如下:

在这里插入图片描述

然而,通过以下代码片段,可以按不同的顺序进行相同的计算,但结果是一样的:

在这里插入图片描述

合并此代码片段与算法的其余部分后,第一个循环体变成了算法中第一个循环体的最后一行。

5.4 原始版本

递归牛顿-欧拉算法最初是由Luh等人(1980a)在一篇论文中描述的,该论文在Brady等人(1982)中再次发表。这个最初的算法是使用3D向量来构成的,表面上看起来与5.3节中描述的算法有很大不同。然而,这两种算法实际上几乎是相同的,唯一的显著区别是其中一个使用空间加速度,而另一个使用经典加速度(见§2.11)。

表5.2显示了原始算法的一个特殊情况,即所有关节都是旋转关节。使用3D向量来表达刚体动力学的一个缺点是方程对关节类型敏感。该算法原始方程式:
ω i = i E i − 1 ω i − 1 + z q ˙ i v i = i E i − 1 ( v i − 1 + ω i − 1 × i − 1 r i ) ω ˙ i = i E i − 1 ω ˙ i − 1 + z q ¨ i + ( i E i − 1 ω i − 1 ) × z q ˙ i v ˙ i = i E i − 1 ( v ˙ i − 1 + ω ˙ i − 1 × i − 1 r i + ω i − 1 × ω i − 1 × i − 1 r i ) F i = m i ( v ˙ i + ω ˙ i × c i + ω i × ω i × c i ) N i = I i c m ω ˙ i + ω i × I i c m ω i f i = F i + i E i + 1 f i + 1 n i = N i + c i × F i + i E i + 1 n i + 1 + i r i + 1 × i E i + 1 f i + 1 τ i = z T n i \begin{aligned} \boldsymbol{\omega}_{i} & ={ }^{i} \boldsymbol{E}_{i-1} \boldsymbol{\omega}_{i-1}+\boldsymbol{z} \dot{q}_{i} \\ \boldsymbol{v}_{i} & ={ }^{i} \boldsymbol{E}_{i-1}\left(\boldsymbol{v}_{i-1}+\boldsymbol{\omega}_{i-1} \times{ }^{i-1} \boldsymbol{r}_{i}\right) \\ \dot{\boldsymbol{\omega}}_{i} & ={ }^{i} \boldsymbol{E}_{i-1} \dot{\boldsymbol{\omega}}_{i-1}+\boldsymbol{z} \ddot{q}_{i}+\left({ }^{i} \boldsymbol{E}_{i-1} \boldsymbol{\omega}_{i-1}\right) \times \boldsymbol{z} \dot{q}_{i} \\ \dot{\boldsymbol{v}}_{i} & ={ }^{i} \boldsymbol{E}_{i-1}\left(\dot{\boldsymbol{v}}_{i-1}+\dot{\boldsymbol{\omega}}_{i-1} \times{ }^{i-1} \boldsymbol{r}_{i}+\boldsymbol{\omega}_{i-1} \times \boldsymbol{\omega}_{i-1} \times{ }^{i-1} \boldsymbol{r}_{i}\right) \\ \boldsymbol{F}_{i} & =m_{i}\left(\dot{\boldsymbol{v}}_{i}+\dot{\boldsymbol{\omega}}_{i} \times \boldsymbol{c}_{i}+\boldsymbol{\omega}_{i} \times \boldsymbol{\omega}_{i} \times \boldsymbol{c}_{i}\right) \\ \boldsymbol{N}_{i} & =\boldsymbol{I}_{i}^{c m} \dot{\boldsymbol{\omega}}_{i}+\boldsymbol{\omega}_{i} \times \boldsymbol{I}_{i}^{c m} \boldsymbol{\omega}_{i} \\ \boldsymbol{f}_{i} & =\boldsymbol{F}_{i}+{ }^{i} \boldsymbol{E}_{i+1} \boldsymbol{f}_{i+1} \\ \boldsymbol{n}_{i} & =\boldsymbol{N}_{i}+\boldsymbol{c}_{i} \times \boldsymbol{F}_{i}+{ }^{i} \boldsymbol{E}_{i+1} \boldsymbol{n}_{i+1}+{ }^{i} \boldsymbol{r}_{i+1} \times{ }^{i} \boldsymbol{E}_{i+1} \boldsymbol{f}_{i+1} \\ \tau_{i} & =\boldsymbol{z}^{\mathrm{T}} \boldsymbol{n}_{i} \end{aligned} ωiviω˙iv˙iFiNifiniτi=iEi1ωi1+zq˙i=iEi1(vi1+ωi1×i1ri)=iEi1ω˙i1+zq¨i+(iEi1ωi1)×zq˙i=iEi1(v˙i1+ω˙i1×i1ri+ωi1×ωi1×i1ri)=mi(v˙i+ω˙i×ci+ωi×ωi×ci)=Iicmω˙i+ωi×Iicmωi=Fi+iEi+1fi+1=Ni+ci×Fi+iEi+1ni+1+iri+1×iEi+1fi+1=zTni

符号:

ω i , ω ˙ i  angular velocity and acceleration of body  i v i , v ˙ i  linear velocity and acceleration of the origin of  F i  (body  i  coordinate frame)  F i , N i  net force and moment on body  i , expressed at its   centre of mass  f i , n i  force and moment exerted on body  i  through joint  i m i , c i , I i c m  mass, centre of mass, and rotational inertia about   centre of mass for body  i  in  F i  coordinates  i E i − 1 3 × 3  orthogonal rotation matrix transforming from  F i − 1  to  F i  coordinates  i − 1 r i  position of origin of  F i  relative to  F i − 1 , expressed in  F i − 1  coordinates  q ˙ i , q ¨ i , τ i  joint  i  velocity, acceleration and force variables  z  direction of joint  i  axis in  F i  coordinates  \begin{aligned} & \boldsymbol{\omega}_{i}, \dot{\boldsymbol{\omega}}_{i} \quad \text { angular velocity and acceleration of body } i \\ & \boldsymbol{v}_{i}, \dot{\boldsymbol{v}}_{i} \quad \text { linear velocity and acceleration of the origin of } F_{i} \\ & \text { (body } i \text { coordinate frame) } \\ & \boldsymbol{F}_{i}, \boldsymbol{N}_{i} \quad \text { net force and moment on body } i \text {, expressed at its } \\ & \text { centre of mass } \\ & \boldsymbol{f}_{i}, \boldsymbol{n}_{i} \quad \text { force and moment exerted on body } i \text { through joint } i \\ & m_{i}, \boldsymbol{c}_{i}, \boldsymbol{I}_{i}^{c m} \quad \text { mass, centre of mass, and rotational inertia about } \\ & \text { centre of mass for body } i \text { in } F_{i} \text { coordinates } \\ & { }^{i} \boldsymbol{E}_{i-1} \quad 3 \times 3 \text { orthogonal rotation matrix transforming from } \\ & F_{i-1} \text { to } F_{i} \text { coordinates } \\ & { }^{i-1} \boldsymbol{r}_{i} \quad \text { position of origin of } F_{i} \text { relative to } F_{i-1} \text {, expressed in } \\ & F_{i-1} \text { coordinates } \\ & \dot{q}_{i}, \ddot{q}_{i}, \tau_{i} \quad \text { joint } i \text { velocity, acceleration and force variables } \\ & \boldsymbol{z} \quad \text { direction of joint } i \text { axis in } F_{i} \text { coordinates } \end{aligned} ωi,ω˙i angular velocity and acceleration of body ivi,v˙i linear velocity and acceleration of the origin of Fi (body i coordinate frame) Fi,Ni net force and moment on body i, expressed at its  centre of mass fi,ni force and moment exerted on body i through joint imi,ci,Iicm mass, centre of mass, and rotational inertia about  centre of mass for body i in Fi coordinates iEi13×3 orthogonal rotation matrix transforming from Fi1 to Fi coordinates i1ri position of origin of Fi relative to Fi1, expressed in Fi1 coordinates q˙i,q¨i,τi joint i velocity, acceleration and force variables z direction of joint i axis in Fi coordinates 

与表5.1中的版本对应关系:

v ^ i = [ ω i v i ] a ^ i = [ ω ˙ i v ˙ i − ω i × v i ] f ^ i B = [ N i + c i × F i F i ] f ^ i = [ n i f i ] i X i − 1 = [ i E i − 1 0 − i E i − 1 i − 1 r i × i E i − 1 ] I ^ i = [ I i c m − m i c i × c i × m i c i × − m i c i × m i 1 ] S i = [ z 0 ] z = [ 0 0 1 ] λ ( i ) = i − 1 μ ( i ) = { i + 1 } \begin{aligned} & \hat{\boldsymbol{v}}_{i}=\left[\begin{array}{c} \boldsymbol{\omega}_{i} \\ \boldsymbol{v}_{i} \end{array}\right] \quad \hat{\boldsymbol{a}}_{i}=\left[\begin{array}{c} \dot{\boldsymbol{\omega}}_{i} \\ \dot{\boldsymbol{v}}_{i}-\boldsymbol{\omega}_{i} \times \boldsymbol{v}_{i} \end{array}\right] \quad \hat{\boldsymbol{f}}_{i}^{B}=\left[\begin{array}{c} \boldsymbol{N}_{i}+\boldsymbol{c}_{i} \times \boldsymbol{F}_{i} \\ \boldsymbol{F}_{i} \end{array}\right] \quad \hat{\boldsymbol{f}}_{i}=\left[\begin{array}{c} \boldsymbol{n}_{i} \\ \boldsymbol{f}_{i} \end{array}\right] \\ & { }^{i} \boldsymbol{X}_{i-1}=\left[\begin{array}{cc} { }^{i} \boldsymbol{E}_{i-1} & \mathbf{0} \\ -{ }^{i} \boldsymbol{E}_{i-1}{ }^{i-1} \boldsymbol{r}_{i} \times & { }^{i} \boldsymbol{E}_{i-1} \end{array}\right] \quad \hat{\boldsymbol{I}}_{i}=\left[\begin{array}{cc} \boldsymbol{I}_{i}^{c m}-m_{i} \boldsymbol{c}_{i} \times \boldsymbol{c}_{i} \times & m_{i} \boldsymbol{c}_{i} \times \\ -m_{i} \boldsymbol{c}_{i} \times & m_{i} \mathbf{1} \end{array}\right] \\ & \boldsymbol{S}_{i}=\left[\begin{array}{l} \boldsymbol{z} \\ \mathbf{0} \end{array}\right] \quad \boldsymbol{z}=\left[\begin{array}{l} 0 \\ 0 \\ 1 \end{array}\right] \quad \lambda(i)=i-1 \quad \mu(i)=\{i+1\} \end{aligned} v^i=[ωivi]a^i=[ω˙iv˙iωi×vi]f^iB=[Ni+ci×FiFi]f^i=[nifi]iXi1=[iEi1iEi1i1ri×0iEi1]I^i=[Iicmmici×ci×mici×mici×mi1]Si=[z0]z= 001 λ(i)=i1μ(i)={i+1}

表5.2:根据Luh等人(1980a)描述的原始递归牛顿-欧拉方程适用于旋转和平移关节,其中包含以下形式的方程式…

(注:接下来应该是具体的方程式形式,但由于你的输入不完整,无法提供详细内容。)
 variable  = {  one expression   if joint is revolute   another expression   if joint is prismatic.  \text { variable }= \begin{cases}\text { one expression } & \text { if joint is revolute } \\ \text { another expression } & \text { if joint is prismatic. }\end{cases}  variable ={ one expression  another expression  if joint is revolute  if joint is prismatic. 

本表中选择使用 3 D 3D 3D向量符号是为了在本书和其他列出该算法的书籍之间达成折衷。该表中的方程是针对一个无分支的运动树编写的,与其他地方描述该算法的方式一致,但很容易将其修改为适用于分支树。

表5.2还显示了原始算法中的 3 D 3D 3D向量与表5.1中显示的空间向量之间的对应关系。只有两个地方 3 D 3D 3D向量与空间向量的对应的 3 D 3D 3D分量不匹配:线性加速度和净物体力矩。后者是一个微不足道的差异,当使用 N i \boldsymbol{N}_{i} Ni来计算 n i \boldsymbol{n}_{i} ni时,它会立即得到纠正。这是由于 N i \boldsymbol{N}_{i} Ni是在质心处计算而不是在物体坐标系的原点处计算所导致的。前者是一个非微不足道的差异,因为它会影响从一个物体传播到下一个物体的量。

(根据2.11节,经典加速度是在一个坐标系中,该坐标系具有纯线性速度,等于固定于物体原点的点的速度,空间速度的明显导数。根据这一点,可以将两个算法之间的差异解释如下:表5.1中的算法在与当前时刻的物体坐标系重合的静止坐标系中表示方程;而表5.2中的算法在与当前时刻的物体坐标系重合、且与物体坐标系原点具有相同线性速度的非旋转坐标系中表示方程。)

原始算法的一个有趣特点是线性速度在任何后续表达式中都没有被使用,因此不需要进行计算。这是使用经典加速度产生的特殊属性。如果省略速度计算,那么该算法比空间版本稍微更高效。然而,这种差异仅仅是几个百分点,并且在其他与效率相关的问题上很容易被忽略,比如编程语言的选择、计算机硬件等等。

5.5 附加注释

Luh等人(1980a)提出的算法通常被称为递归牛顿-欧拉算法。然而,将牛顿-欧拉运动方程与递归评估策略相结合的想法在几年前就有了。特别是,在Stepanenko和Vukobratovic(1976)以及Orin等人(1979)的早期工作中可以看到递归牛顿-欧拉算法的初步形式。该算法随后经过改进和优化,其中一些优化方法在第10章中进行了介绍。最快的两个已发表版本可以在He和Goldenberg(1989)以及Balafoutis等人(1988)中找到,后者还在Balafoutis和Patel(1991)中提到。当时机器人技术领域进行这类研究的动力是为了开发一种逆动力学算法,以便在实时控制机器人运动中能够快速使用。Luh等人(1980b)中可以找到一个早期的例子,尽管最终操作空间的形式化变得更受欢迎(Khatib,1987)。当时计算机的相对较慢速度使得一些研究人员开始研究并行计算机实现的可能性(Lathrop,1985),而其他人则研究符号优化技术(Murray和Neuman,1984;Neuman和Murray,1987)。有一个有趣的新方向是开发高效的算法来计算相对于设计参数的逆动力学的偏导数。Fang和Pollard(2003)提供了一个例子。

在该算法被发明的时候,拉格朗日动力学比牛顿-欧拉动力学更受欢迎,这促使Hollerbach设计了递归拉格朗日算法,并比较了各种递归和非递归算法的计算复杂性(Hollerbach,1980)。这项工作对比了非递归方法(如Uicker/Kahn算法(Kahn和Roth,1971))的 O ( n 4 ) O\left(n^{4}\right) O(n4)复杂性与递归算法的 O ( n ) O(n) O(n)复杂性。

  • 23
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值