肌肉骨骼模拟

2024-01-30 03:52
文章标签 模拟 骨骼 肌肉

本文主要是介绍肌肉骨骼模拟,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

对于角色动画,目前的技术主要包括这两种方式:基于运动学模拟的动画,和基于动力学模拟、即基于物理模拟的动画。第一种方式的应用相当广泛,我们接触的游戏大部分都采用了基于运动学的方式去实现动画,这种方式性能优秀,便于实现,然而却缺少了许多真实性。基于物理模拟的方式来实现动画,对性能的要求比较高,且实现起来比较复杂,优点是交互效果更真实。对人体的肌肉骨骼进行模拟,属于物理模拟的动画实现。

肌肉的模拟较为复杂,涉及到生物力学这门交叉学科的内容。目前较为通用的肌肉模拟的模型是希尔模型(Hill model)。接下来我会根据一篇2013年的TOG论文对肌肉模型的模拟进行介绍。

本篇博客参考自《Flexible Muscle-Based Locomotion for Bipedal Creatures》。

肌肉骨骼模型

希尔模型将肌肉表示成三元素结构,它由下列要素组成:

  • 一个收缩单元(contractile element, CE),表示根据肌肉激活状态收缩的肌肉纤维,这个单元可以产生主动力。
  • 一个并行弹性单元(parallel elastic element, PEE),表示肌肉纤维周围的被动弹性组织,这个单元可以产生被动力。
  • 一个串行弹性单元(parallel elastic element, SEE),代表连接肌肉和骨骼的肌腱,这个单元可以产生被动力。
    在这里插入图片描述

CE产生的力 F C E F_{CE} FCE,取决于肌肉的恒定最大等距力 F m a x F_{max} Fmax、肌肉激活 a a a、肌肉纤维的长度 L C E L_{CE} LCE,以及收缩速度 V C E V_{CE} VCE
F C E = a F m a x f L ( L C E ) f V ( V C E ) \begin{align} F_{CE} = aF_{max}f_L(L_{CE})f_V(V_{CE}) \end{align} FCE=aFmaxfL(LCE)fV(VCE)
其中, f L f_L fL表示力与肌肉长度之间的关系, f V f_V fV表示力与当前收缩速度之间的关系,下图即为这两个函数:

在这里插入图片描述

弹性元件PEE和SEE产生的被动力 F P E E F_{PEE} FPEE F S E E F_{SEE} FSEE根据其长度可以建模为非线性弹簧:
F S E E = f S E E ( L M − L C E ) F P E E = f P E E ( L C E ) \begin{align} F_{SEE} &= f_{SEE}(L_M-L_{CE}) \\ F_{PEE} &= f_{PEE}(L_{CE}) \end{align} FSEEFPEE=fSEE(LMLCE)=fPEE(LCE)
其中 f S E E f_{SEE} fSEE f P E E f_{PEE} fPEE是非线性的力-长度关系, L M L_M LM为肌肉的总长度。

由于SEE与CE和PEE串联在一起,因此总肌肉力 F M F_M FM服从力平衡方程:
F M = F C E + F P E E = F S E E \begin{align} F_M = F_{CE} + F_{PEE} = F_{SEE} \end{align} FM=FCE+FPEE=FSEE
SEE的初始化长度为其最优长度, L C E o p t L_{CE}^{opt} LCEopt。结合肌腱的松弛长度 L S E E s l a c k L_{SEE}^{slack} LSEEslack,可以定义肌肉的休息长度 L M r e s t L_M^{rest} LMrest:
L M r e s t = L S E E s l a c k + L C E o p t \begin{align} L_M^{rest} = L_{SEE}^{slack} + L_{CE}^{opt} \end{align} LMrest=LSEEslack+LCEopt
在实际进行模拟的时候,输入参数为激活状态 a a a以及肌肉的总长度 L M L_M LM

激活状态的动态更新

肌肉的激活状态 a a a是一个相对缓慢的电化学过程的结果,该过程基于控制系统输出的神经兴奋信号 u u u。这个过程称为激活动力学,其模型为:
∂ a ∂ t = c a ( a − u ) \begin{align} \frac{\partial a}{\partial t} = c_a(a-u) \end{align} ta=ca(au)
其中 c a c_a ca是恒定的激活和失活速率。

肌肉几何以及肌肉与骨骼的相互作用

骨骼和肌肉具有双向的作用:肌肉施加改变骨骼姿势的力量,而骨骼姿势决定肌肉的长度 L M L_M LM,进而影响收缩动力。整个肌肉的路径是由它的肌腱附着在骨头上的位置、肌肉环绕的骨性标志以及随时间变化的肌肉跨越的关节决定的。在原文的模型中,将肌肉路径定义为一组线段。这个模型是一种简化,具有高性能的优势,并且省略了对肌肉的几何形状建模的需要。如下图,长方体代表着肌肉附着的骨骼,红色连线代表着对肌肉的简化:

在这里插入图片描述

任何肌肉 M M M​的路径都是由 n n n​个附着点数组定义的, [ { b 1 , p 1 } . . . , { b n , p n } ] \left[ \left\{ b_1, p_1 \right\}...,\left\{b_n,p_n\right\} \right] [{b1,p1}...,{bn,pn}],每一个附着点由一个偏移量 p i \mathbf p_i pi以及一个附着在的骨骼 b i b_i bi定义。偏移量 p i \mathbf p_i pi定义为物体 b i b_i bi坐标系中的一个固定偏移量,它和它附着着的骨骼一起移动和旋转。多个附着点可以附着在一个骨骼上。第一个点和最后一个点 p 1 p_1 p1 p n p_n pn表示肌肉肌腱附着在骨骼上的位置,其他点都是通过点。

肌肉的总长度 L M L_M LM等于 n − 1 n - 1 n1个肌肉段长度之和, [ s 1 , . . . , s n − 1 ] [s_1,...,s_{n-1}] [s1,...,sn1],这些肌肉段的长度由世界坐标系中每个点的位置 p i W p_i^W piW求出:
L M = ∑ i = 1 n − 1 ∣ ∣ s i ∣ ∣ , s i = p i + 1 W − p i W \begin{align} L_M = \sum_{i=1}^{n-1} || s_i ||, s_i = p_{i+1}^W - p_i^W \end{align} LM=i=1n1∣∣si∣∣,si=pi+1WpiW
这些附着点在这篇文章的定义中,可以有四种活动方式:要么是固定的,要么被限制在一条线、一个平面或一个盒子上:

在这里插入图片描述

肌肉如何产生作用

由肌肉总长度 L M L_M LM以及激活状态 a a a可以计算肌肉收缩力 F M F_M FM。由于受力平衡,这个力被传递到肌肉在骨骼上的每个附着点上,并在它跨越的每个关节上产生扭矩。对于每个关节 k k k,在力臂 r k r_k rk定义的方向上可以产生一个扭矩 τ k \tau_k τk。这个力臂对应于穿过关节的肌肉段方向 s c s_c sc与关节中心 j k j_k jk到附着点 p c W p_c^W pcW的矢量之间的叉积:
τ k = F M ∣ ∣ r k ∣ ∣ , r k = ( p c W − j k ) × s c ∣ ∣ s c ∣ ∣ \begin{align} \tau_k = F_M||r_k||, r_k = (p_c^W - j_k) \times \frac{s_c}{||s_c||} \end{align} τk=FM∣∣rk∣∣,rk=(pcWjk)×∣∣sc∣∣sc

在Unity中实现肌肉骨骼模拟

在unity资源商店中有一个资源,实现了上述的肌肉骨骼模型:

https://assetstore.unity.com/packages/tools/physics/kinesis-physical-muscle-model-based-movement-206089

比较贵,$69.99,忍痛割刀买了。接下来讲解一下这个资源里是如何实现肌肉骨骼模型的。

在这个资源里,实现了调节肌肉激活状态,驱动人物的关节运动的物理模拟。

底层模拟基础

为了实现添加扭矩,驱动关节移动和旋转的功能,利用unity提供的Rigidbody以及Character Joint组件实现底层的模拟。

模拟实现

在每一个时步的模拟中,首先,参考公式 ( 6 ) (6) (6),根据输入的激活值更新激活状态 a a a。然后,更新SEE部分的长度(如果是初始调用,则不更新)。根据SEE的长度计算归一化后CE的长度,以及归一化后CE的收缩速度(如果是初始调用,则为0)。根据激活状态 a a a、归一化后CE的长度、归一化后CE的收缩速度,计算CE部分产生的主动力,参考公式 ( 1 ) (1) (1)。同时,根据CE的长度,计算PEE部分的被动力。CE部分产生的主动力,加上PEE部分产生的被动力,等于SEE部分产生的被动力,也等于肌肉产生的力,参考公式 ( 4 ) (4) (4)。其中,SEE部分产生的被动力,可以用来更新SEE部分的长度。接下来,利用肌肉产生的力,计算肌肉穿过的关节的扭矩,参考公式 ( 8 ) (8) (8)

在这里插入图片描述

补充

之前在实习时,leader让我做过这样一件事情,根据人物的运动序列,反向计算肌肉的激活状态,当时没能实现。现在想想,可以这样实现:

1、从人物的运动学模型转换到动力学模型,通过PD微分,求解人物每个关节所需要的扭矩。

2、根据每块肌肉和关节的关系,计算出每块肌肉应该产生的发力,通过每块肌肉产生的发力,反求出肌肉的激活状态。(根据公式 ( 1 ) (1) (1),并对激活状态的反推做简化,用 a = F C E / F m a x a = F_{CE} / F_{max} a=FCE/Fmax计算)

参考

Geijtenbeek T, Van De Panne M, Van Der Stappen A F. Flexible muscle-based locomotion for bipedal creatures[J]. ACM Transactions on Graphics (TOG), 2013, 32(6): 1-11.

https://assetstore.unity.com/packages/tools/physics/kinesis-physical-muscle-model-based-movement-206089

这篇关于肌肉骨骼模拟的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/659026

相关文章

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

usaco 1.2 Transformations(模拟)

我的做法就是一个一个情况枚举出来 注意计算公式: ( 变换后的矩阵记为C) 顺时针旋转90°:C[i] [j]=A[n-j-1] [i] (旋转180°和270° 可以多转几个九十度来推) 对称:C[i] [n-j-1]=A[i] [j] 代码有点长 。。。 /*ID: who jayLANG: C++TASK: transform*/#include<

hdu4431麻将模拟

给13张牌。问增加哪些牌可以胡牌。 胡牌有以下几种情况: 1、一个对子 + 4组 3个相同的牌或者顺子。 2、7个不同的对子。 3、13幺 贪心的思想: 对于某张牌>=3个,先减去3个相同,再组合顺子。 import java.io.BufferedInputStream;import java.io.BufferedReader;import java.io.IOExcepti

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟)

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟) 题目描述 给定一个链表,链表中的每个节点代表一个整数。链表中的整数由 0 分隔开,表示不同的区间。链表的开始和结束节点的值都为 0。任务是将每两个相邻的 0 之间的所有节点合并成一个节点,新节点的值为原区间内所有节点值的和。合并后,需要移除所有的 0,并返回修改后的链表头节点。 思路分析 初始化:创建一个虚拟头节点

每日一题|牛客竞赛|四舍五入|字符串+贪心+模拟

每日一题|四舍五入 四舍五入 心有猛虎,细嗅蔷薇。你好朋友,这里是锅巴的C\C++学习笔记,常言道,不积跬步无以至千里,希望有朝一日我们积累的滴水可以击穿顽石。 四舍五入 题目: 牛牛发明了一种新的四舍五入应用于整数,对个位四舍五入,规则如下 12345->12350 12399->12400 输入描述: 输入一个整数n(0<=n<=109 ) 输出描述: 输出一个整数

【算法专场】模拟(下)

目录 前言 38. 外观数列 算法分析 算法思路 算法代码 1419. 数青蛙 算法分析 算法思路 算法代码  2671. 频率跟踪器 算法分析 算法思路 算法代码 前言 在前面我们已经讲解了什么是模拟算法,这篇主要是讲解在leetcode上遇到的一些模拟题目~ 38. 外观数列 算法分析 这道题其实就是要将连续且相同的字符替换成字符重复的次数+

模拟实现vector中的常见接口

insert void insert(iterator pos, const T& x){if (_finish == _endofstorage){int n = pos - _start;size_t newcapacity = capacity() == 0 ? 2 : capacity() * 2;reserve(newcapacity);pos = _start + n;//防止迭代

PHP实现二叉树遍历(非递归方式,栈模拟实现)

二叉树定义是这样的:一棵非空的二叉树由根结点及左、右子树这三个基本部分组成,根据节点的访问位置不同有三种遍历方式: ① NLR:前序遍历(PreorderTraversal亦称(先序遍历)) ——访问结点的操作发生在遍历其左右子树之前。 ② LNR:中序遍历(InorderTraversal) ——访问结点的操作发生在遍历其左右子树之中(间)。 ③ LRN:后序遍历(PostorderT

1 模拟——67. 二进制求和

1 模拟 67. 二进制求和 给你两个二进制字符串 a 和 b ,以二进制字符串的形式返回它们的和。 示例 1:输入:a = "11", b = "1"输出:"100"示例 2:输入:a = "1010", b = "1011"输出:"10101" 算法设计 可以从低位到高位(从后向前)计算,用一个变量carry记录进位,如果有字符没处理完或者有进位,则循环处理。两个字符串对

AMAZING AUCTION(简单模拟)

AMAZING AUCTION 时间限制: 3000 ms  |  内存限制: 65535 KB 难度:4 描述 Recently the auction house hasintroduced a new type of auction, the lowest price auction. In this new system,people compete for the lo