BlendTree动画混合算法详解

news2025/1/11 14:12:03

【混合本质】

如果了解骨骼动画就知道,某一时刻角色的Pose是通过两个邻近关键帧依次对所有骨骼插值而来,换句话说就是由两个关键帧混合而来。

那么可不可以由多个关键帧混合而来呢?当然可以。

更多的关键帧可以来自不同的动画片段(AnimationClip),混合时给予每个动画片段的权重不同,但权重之和仍然为1。

我们可以预先指定混合时需要哪些动画片段,通过混合算法在运行时计算每个动画片段的权重。

通过混合可以用有限个动画片段混合出多种角色状态,进而减少动画片段的制作。

混合一般用于角色运动locomotion。

【一维线性插值混合】

一维线性插值混合要求不同动画片段只有一个属性不同,例如角色运动的速度和朝向不同。

以速度为例,有五个动画片段,速度分别是1 2 3 4 5m/s,将不同动画片段按照属性的大小从左到右排成一行。

那么该BlendTree有五个AnimationClip,记录了每个AnimationClip的属性,自身有一个属性参数speed。

运行时角色的运动速度会变化,角色速度的值即为该BlendTree的Speed属性当前的值。

可以从五个AnimationClip中找到当前速度邻近的两个,这样就找到了运行时需要参与混合的动画片段。

(注意,当前速度小于动画片段的最小值,或者大于最大值,那么邻近的动画片段只有1个了,不用混合了)

接下来需要确定两个动画片段的权重分别是多少。显然根据两个动画片段的速度值和当前的速度值,可以确定不同动画的权重。(如何这里看不懂怎么确定的,请先了解下线性插值)

【二维双线性插值混合】

双线性插值需要四个点,Q11、Q21、Q12、Q22。

有一个插值系数a1,让Q11和Q21完成线性插值得到R1,Q12和Q22同样用a2完成线性插值得到R2。

有另一个插值系数a2,让R1和R2完成线性插值得到P

在动画混合中,四个点就表示四个动画片段,X轴和Y轴表示角色运动的两个属性,以速度和朝向为例,a1和a2是运行时根据角色当前的速度和朝向,以及动画片段的属性计算出来的。这个坐标系叫做参数空间parameter space

其中,因为存在公用a1和a2的问题,假设a1对应速度属性,a2对应朝向属性。则要求动画片段Q11和Q12、Q21和Q22相互间速度属性不同,Q11和Q12、Q21和Q22相互间朝向属性不同。

也即,两个属性的两个不同值,对应4个动画片段。

如果两个属性分别由N、M个不同的值,则对应N*M个动画片段。

在实际运行时,需要先根据属性值知道四个动画片段,分两次找,每次找的方式与一维线性插值混合找的方式相同,然后进行双线性插值,得到最后的角色Pose

【二维三角形插值混合】

线性插值是用两个已知点插值计算出一个未知点。当然也可以用三个已知点插值计算出一个未知点。

y = a * y1 + b* y2 + c* y3。一般的,a + b + c = 1.

因此,要计算出插值结果,需要先算出两个未知数a和b。

这恰好可以通过三角形的重心坐标算出:二维坐标系中三个已知点可以组合出来三个向量,向量点乘算出面积,面积之比算出未知数。

在动画混合中,三个点代表三个不同的动画片段,点的坐标是动画片段的属性值。

如果有很多个动画呢?如同二维双线性插值混合中的做法,先找到要参与混合的三个动画片段。

如何找?

我们通常假设在坐标原点附近(最好就是在坐标原点)有一个中心动画片段,这个片段一般为类似Idle的动画片段,叫做中心点。

中心点和所有其他点(也叫采样点)都有一个连线,通过判断运行时传过来的两个属性值构成的点(也叫目标点)位于哪两条连线形成的夹角之间即可。

如何判断?向量叉乘可以判断点在直线的左侧还是右侧。

【更复杂的插值混合】

上述三种混合算法都是比较容易想到、容易理解的算法,真正进行混合时,实际参与混合的动画片段为2或3个。如何希望有更多的动画片段参与混合,那么如何给定每个参数的权重值?

这实质上是一个离散数据插值问题,这里有一个基本假设,假设某个采样点对目标点的影响因子为h,所有采样点影响和为H,那么采样点的权重w = h/H。

注意如果H = 1,那么影响因子就是权重了。

我们只需要找到如何计算影响因子h即可,前面三种简单的方式也是如此,插值方法如下:

  • 反距离权重插值Inverse Distance Weighted Interpolation
  • 自然邻近插值Natural Neighbors Interpolation
  • 径向基函数插值Radial Basis Function Interpolation
  • K邻近插值K-Nearest-Neighbors Interpolation
  • 梯度带插值Gradient Band Interpolation

【反距离权重插值】

计算所有采样点到目标点的距离,影响因子为距离的倒数 h= 1/distance(s,p)

可以看到距离目标点越近的采样点影响因子越大。

为了减少计算量,可以

1.限制有点的影响距离D,超过该距离,认为不会产生影响

2.不算距离,取距离的平方

【自然邻近插值】

自然邻近插值基于面积(area-based),首先要算出每个采样点的影响面积,采用维诺图算法计算。

随后计算加入目标点后,目标点与周围点形成的维诺图

目标点维诺图和周围点维诺图的重叠面积即为影响因子,如图所示:

这种方式计算量大,不适合实时计算。

【径向基函数插值】

学过线性代数,我们知道,在向量空间中,任意一个向量都可以用一组基向量的线性组合表示,二维向量空间的基向量个数为2,三维向量空间的基向量个数为3,N维向量空间的基向量个数为N

在函数空间中,任意一个函数都可以用一组基函数的线性组合表示。

径向基函数是取值仅依赖于到原点距离的实值函数。仅依赖表示函数只有一个自变量,为该点到原点的距离,注意原点不是坐标系的原点,原点是我们自己选的。到原点的距离一般指欧式距离,也即我们常用的计算两个点距离的方法。

基本的径向基函数有:

在动画混合中,目标点就是原点,根据其他点到原点的距离以及径向基函数可以计算出来影响因子的值。

【K邻近插值】

类似反距离权重插值,这里只选择距离目标点最近的K个采样点参与实际的混合,第K个采样点的权重为0,更远的采样点的权重也为0。影响因子的计算与反距离权重插值相同。

【梯度带插值】

有三个点O、A、B,在一维线性插值混合中,他们在一条直线上,通过判断O点到A点、O点到B点的距离来计算每个点的影响因子,那么A点和B点的影响因子分别为(此时权重值等于影响因子的值)

h(A) =1 - OA/AB   h(B) =1 - OB/AB。

注意如果O在A点左边,那么A的影响因子为1,B的影响因子为0。反之亦然。

如果O点不在AB所在的直线呢?

这时计算距离,要多一个O到AB所在直线的距离。这一项对A点和B点一样,可以抵消掉。我们只要得到O点分别到A点和B点的水平距离即可,用同样的公式可以得到A和B对O的影响。

这个距离如何求?可以看到这实际上是在求向量OA在向量AB上的投影长度。(不知道怎么计算投影长度可以先学习下),随后可以得到h(A)、h(B)

h(A)即为相对于B点,A点对O点的梯度值。如果还有C、D、E点,可以求出相对于这些点,A点对O点的梯度值。

从这些梯度值中,找到最小值,即为在所有采样点中,A点对目标点的影响因子,用公式表示为:

对于已经给定的采样点,如果目标点是未知数,可以计算得到某个采样点关于目标点的梯度值分布(即梯度带),如下图所示:

可以看到P1在紫色三角形的顶点附近,仍然存在一定的影响,对含有速度的混合而言,可能不希望受到P1的影响。

这时需要在计算影响因子时考虑角度。角度时目标点P与某个采样点A所在向量的夹角,夹角越小,影响越大。

同样的,如果原点为O,如果点P在直线OA左侧,采样点B在直线OA右侧,那么A点的权重应该为1。

如何综合距离和角度计算影响因子?参考极坐标系的方式,建立一个速度和角度的坐标系,求出目标点和采样点在该坐标系中的位置,再做基于距离的梯度带差值。也即,修改向量的坐标表示方式,如下图所示,其中α是控制因子。

【动画混合中权重算法的评价标准】

  • 准确性:如果目标点与某个采样点重合,那么其权重必须为1,其他采样点权重为0。实际上,如果目标点靠近采样点的某个小范围内,那么将目标点和采样带你视为重合。这样做的目的是保留动画师愿意的原始动作。而在回归分析中,会通过统计方法估计出一个最佳的拟合值。
  • 累加和:所有权重累加和必须为1
  • 连续性:权重函数要具有C0连续性,目标点小的变化只能使权重的变化小,否则会引起混合动作的抖动
  • 有界性:所有权重都要在0和1之间,否则会混合出不可预测和不理想的结果
  • 局部性:只有目标的附近一定范围内的采样点的权重会大于0,其他采样点的权重为0,这可以提高混合效率。
  • 单调性: 每个权重函数应该从其全局最大值递减到全局最小值。严格来说,权重函数不应该有局部最小值。例如,一个奔跑动作的权重应该随着速度从奔跑速度降低到行走速度而从1递减到0。
  • 密度不变性:对分布均匀和不均匀的采样点有同样的效果,不应该对附近采样点多的地方比同样邻近的单个采样点更大的权重

【参考】

Unity - Manual: 2D Blending

《游戏引擎架构》

Automated
Semi‐Procedural
Animation for
Character
Locomotion

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1203673.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【MySQL】事务(中)

文章目录 事务异常与产出结论手动提交 和自动提交 对 回滚的区别 事务隔离性理论如何理解隔离性?MySQL的隔离级别事务隔离级别的查看设置隔离级别 事务异常与产出结论 在没有启动事务之前,account表中存在孙权和刘备的数据 在启动事务后, 向 …

通过 Kaptcha 插件生成字符验证码

Kaptcha 是 Google 的⼀个⾼度可配置的实⽤验证码⽣成⼯具,我们选择的是⼀个适配SpringBoot的 开源项⽬ 生成的验证码效果如下: 原理 验证码可以客户端生成,也可以服务器生成. 对于普通的字符验证码, 后端通常分两部分: ⼀:⽣成验…

C语言求数组中出现次数最多的元素

一、前言 遇到一个需求,需要求数组中出现次数最多的元素,查找了一些资料,结合自己的思路,编写了程序并验证。 只考虑元素为非负整数的数组,如果有出现次数相同的元素,则返回较小元素。 二、编程思路 以数…

原生JS实现视频截图

视频截图效果预览 利用Canvas进行截图 要用原生js实现视频截图,可以利用canvas的绘图功能 ctx.drawImage,只需要获取到视频标签,就可以通过drawImage把视频当前帧图像绘制在canvas画布上。 const video document.querySelector(video) con…

若依系统富文本框上传图片报错!

报错如下: 原因:如图,富文本路径中存在 / 字符,导致上传出错。 解决方案:将富文本框内容在前端进行加密,后端再解密。 前端: 安装 crypto-js 插件 npm install crypto-js 创建工具类 :在 sr…

postswigger 靶场(CSRF)攻略-- 1.没有防御措施的 CSRF 漏洞

靶场地址: What is CSRF (Cross-site request forgery)? Tutorial & Examples | Web Security Academy (portswigger.net)https://portswigger.net/web-security/csrf 没有防御措施的 CSRF 漏洞 题目中已告知易受攻击的是电子邮件的更改功能,而目…

【正点原子STM32连载】 第五十二章 图片显示实验摘自【正点原子】APM32F407最小系统板使用指南

1)实验平台:正点原子stm32f103战舰开发板V4 2)平台购买地址:https://detail.tmall.com/item.htm?id609294757420 3)全套实验源码手册视频下载地址: http://www.openedv.com/thread-340252-1-1.html## 第五…

c语言从入门到实战——初识指针

初识指针 前言1. 内存和地址1.1 内存1.2 究竟该如何理解编址 2. 指针变量和地址2.1 取地址操作符(&)2.2 指针变量和解引用操作符(*)2.2.1 指针变量2.2.2 如何拆解指针类型2.2.3 解引用操作符 2.3 指针变量的大小 3. 指针变量类…

Sprint Boot 学习路线 6

测试 Spring提供了一组测试工具,可以轻松地测试Spring应用程序的各个组件,包括控制器、服务、存储库和其他组件。它具有丰富的测试注释、实用程序类和其他功能,以帮助进行单元测试、集成测试等。 JPA测试 Spring JPA(Java Pers…

开放领域对话系统架构

开放领域对话系统是指针对非特定领域或行业的对话系统,它可以与用户进行自由的对话,不受特定领域或行业的知识和规则的限制。开放领域对话系统需要具备更广泛的语言理解和生成能力,以便与用户进行自然、流畅的对话。 与垂直领域对话系统相比…

从单服务设计看SLA保证

文章首发公众号:海天二路搬砖工 0. 引言 在微服务架构中,谈到SLA保证,我们更多是从宏观的角度来需求解决方案。比如,通过合理服务拆分来增加系统整体的可维护性;通过多实例部署来保证系统的灾备。但是单个服务是可靠…

2023NewStarCTF

目录 一、阳光开朗大男孩 二、大怨种 三、2-分析 四、键盘侠 五、滴滴滴 六、Include? 七、medium_sql 八、POP Gadget 九、OtenkiGirl 一、阳光开朗大男孩 1.题目给出了secret.txt和flag.txt两个文件,secret.txt内容如下: 法治自由公正爱国…

【JVM】类加载器 Bootstrap、Extension、Application、User Define 以及 双亲委派

以下环境为 jdk1.8 两大类 分类成员语言继承关系引导类加载器bootstrap 引导类加载器C/C无自定义类加载器extension 拓展类加载器、application 系统/应用类加载器、user define 用户自定义类加载器Java继承于 java.lang.ClassLoader 四小类 Bootstrap 引导类加载器 负责加…

故障演练 | 微服务架构下如何做好故障演练

前言 微服务架构场景中,应用系统复杂切分散。长期运行时,局部出现故障时不可避免的。如果发生故障时不能进行有效反应,系统的可用性将极大地降低。 什么是故障演练 故障演练是指模拟生产环境中可能出现的故障,测试系统或应用在…

软考网络工程师知识点总结(二)

目录 21、海明码--差错控制 22、CRC循环冗余校验码 23、网络时延的计算 24、根据距离选择传输介质 25、多模光纤和单模光纤的区别 26、CSMA/CD协议 27、以太网帧结构 28、以太网类型及传输介质的选择 29、交换式以太网(交换机) 30、VLAN虚拟局…

Rust编程中的线程间通信

1.消息传递 为了实现消息传递并发,Rust 标准库提供了一个 信道(channel)实现。信道是一个通用编程概念,表示数据从一个线程发送到另一个线程。 可以将编程中的信道想象为一个水流的渠道,比如河流或小溪。如果你将诸如…

【C++ 学习 ㊱】- 智能指针详解

目录 一、为什么需要智能指针? 二、智能指针的原理及使用 三、auto_ptr 3.1 - 基本使用 3.2 - 模拟实现 四、unique_ptr 4.1 - 基本使用 4.2 - 模拟实现 五、shared_ptr 5.1 - 基本使用 5.2 - 模拟实现 六、weak_ptr 6.1 - shared_ptr 的循环引用问题 …

【Python小程序】求解2 * 2矩阵的逆矩阵

一、内容简介 使用Python求解2 * 2矩阵的逆矩阵。 二、求解方法 我们使用邻接矩阵法来求解2 * 2矩阵的逆矩阵。 det(A): 矩阵A的行列式 adj(A): 矩阵A的邻接矩阵 对于2*2矩阵A 我们有 三、Python代码 基于上述求解方法,我们可以写出Python代码如下&#xff…

行情分析——加密货币市场大盘走势(11.13)

大饼上涨太快,又开始震荡,但上不去,所以目前来看差不多要做回踩动作,入场空单性价比较高。而且从MACD日线来看,也是进入空头趋势,RSI(14)也是进入了超买区间,值得入手空单…