主要研究人体姿态估计中heatmap转坐标的方法,提出一种新的解码方法
(其实这人体姿态我毛也不会,过来看看这个heatmap解码方法)
代码:https://github.com/ilovepose/DarkPose/blob/master/lib/core/inference.py
方法
人体姿态估计简单来说就是要预测人的所有关节的坐标,这里主要研究把关节转为热体图,然后进行回归的那种方法。
根据上下文推测,这里应该是一个关键点一个通道heatmap
Coordinate Decoding
如果输出的heatmap的尺寸和原图的大小一样,那么我们只需要找到最大的点。
然而通常我们输出的heatmap需要上采样到原来的尺寸,也就是乘以一个系数 λ ∈ R + + \lambda \in \mathbb{R}_{++} λ∈R++
这将导致亚像素定位问题(sub-pixel localisation)
The standard coordinate decoding method
令 h \mathbf{h} h为模型输出的heatmap
heatmap最大值 m \mathbf{m} m,第二大值 s \mathbf{s} s
那么heatmap中的关节位置
p = m + 1 4 s − m ∥ s − m ∥ 2 \mathbf{p} = \mathbf{m} + \frac{1}{4}\frac{\mathbf{s}-\mathbf{m}}{\|\mathbf{s}-\mathbf{m}\|_2} p=m+41∥s−m∥2s−m
这个就是最大值往第二大值的方向移动 1 4 \frac{1}{4} 41像素
最终原图像的关节位置为
p ^ = λ p \hat{\mathbf{p}} = \lambda \mathbf{p} p^=λp
这个移动的主要目的是补偿下采样导致的量化效应
heatmap中最大值点并不代表真实关节的位置,只是一个大概的位置。
从这个表可以看出,这个移动能够带来很大的提升
Our coordinate decoding method
我们假设预测出来的heatmap和生成的heatmap服从高斯分布
因此预测的heatmap
G ( x ; μ , Σ ) = 1 ( 2 π ) ∣ Σ ∣ 1 2 e x p ( − 1 2 ( x − μ ) T Σ − 1 ( x − μ ) ) \mathcal{G}\left(\mathbf{x};\mathbf{\mu}, \mathbf{\Sigma}\right) = \frac{1}{\left(2\pi\right) \left|\mathbf{\Sigma}\right|^{\frac{1}{2}}}\rm{exp}\left(-\frac{1}{2}\left(\mathbf{x}-\mathbf{\mu}\right)^T\mathbf{\Sigma}^{-1}\left(\mathbf{x}-\mathbf{\mu}\right)\right) G(x;μ,Σ)=(2π)∣Σ∣211exp(−21(x−μ)TΣ−1(x−μ))
其中 x \mathbf{x} x是预测heatmap中的像素坐标,
μ \mathbf{\mu} μ是高斯中心,也就是预测的关节位置
Σ \mathbf{\Sigma} Σ是协方差矩阵,是一个对角矩阵
Σ = [ σ 2 0 0 σ 2 ] \mathbf{\Sigma} = \begin{bmatrix} \sigma^2 & 0 \\ 0 & \sigma^2 \end{bmatrix} Σ=[σ200σ2]
σ \sigma σ是标准差
取个对数
P ( x ; μ , Σ ) = ln ( G ) = − ln ( 2 π ) − 1 2 ln ( ∣ Σ ∣ ) − 1 2 ( x − μ ) T \begin{aligned} \mathcal{P}\left(\mathbf{x};\mathbf{\mu},\mathbf{\Sigma}\right) &= \ln \left(\mathbf{G}\right)\\ &=-\ln\left(2\pi\right) - \frac{1}{2}\ln\left(\left|\mathbf{\Sigma}\right|\right)-\frac{1}{2}\left(\mathbf{x}-\mathbf{\mu}\right)^T \end{aligned} P(x;μ,Σ)=ln(G)=−ln(2π)−21ln(∣Σ∣)−21(x−μ)T
一阶导
D ′ ( x ) ∣ x = μ = ∂ P T ∂ x ∣ x = μ = − Σ − 1 ( x − μ ) ∣ x = μ = 0 \left.\mathcal{D}^{\prime}(\boldsymbol{x})\right|_{\boldsymbol{x}=\boldsymbol{\mu}}=\left.\frac{\partial \mathcal{P}^T}{\partial \boldsymbol{x}}\right|_{\boldsymbol{x}=\boldsymbol{\mu}}=-\left.\Sigma^{-1}(\boldsymbol{x}-\boldsymbol{\mu})\right|_{\boldsymbol{x}=\boldsymbol{\mu}}=0 D′(x)∣x=μ=∂x∂PT x=μ=−Σ−1(x−μ) x=μ=0
二阶导
D ′ ′ ( m ) = D ′ ′ ( x ) ∣ x = m = − Σ − 1 \mathcal{D}^{\prime \prime}(\boldsymbol{m})=\left.\mathcal{D}^{\prime \prime}(\boldsymbol{x})\right|_{\boldsymbol{x}=\boldsymbol{m}}=-\Sigma^{-1} D′′(m)=D′′(x)∣x=m=−Σ−1
在最大值点泰勒展开到二阶
P ( μ ) = P ( m ) + D ′ ( m ) T ( μ − m ) + 1 2 ( μ − m ) T D ′ ′ ( m ) ( μ − m ) \mathcal{P}(\boldsymbol{\mu})=\mathcal{P}(\boldsymbol{m})+\mathcal{D}^{\prime}(\boldsymbol{m})^T(\boldsymbol{\mu}-\boldsymbol{m})+\frac{1}{2}(\boldsymbol{\mu}-\boldsymbol{m})^T \mathcal{D}^{\prime \prime}(\boldsymbol{m})(\boldsymbol{\mu}-\boldsymbol{m}) P(μ)=P(m)+D′(m)T(μ−m)+21(μ−m)TD′′(m)(μ−m)
接着就能得到
μ = m − ( D ′ ′ ( m ) ) − 1 D ′ ( m ) \boldsymbol{\mu} = \mathbf{m}-\left(\mathcal{D}^{\prime \prime}(\boldsymbol{m})\right)^{-1}\mathcal{D}^{\prime}(\boldsymbol{m}) μ=m−(D′′(m))−1D′(m)
说实话,不知道这个怎么得到的,我猜测是泰勒展开后求导为0得到的
代码里,一阶导和二阶导是用数字图像处理那种求导方法得到的
Heatmap distribution modulation
这个图(a)里是预测的heatmap,可以看出其实是有多个峰值的
这可能会对作者的解码方法产生负面影响。为了解决这个问题,先调整一下这个heatmap的分布
更具体地,为了解决这种多个峰值的情况,作者使用了和训练时方差一样的高斯核 K \mathbf{K} K
h ′ = K ⊛ h \mathbf{h}^{\prime} = \mathbf{K} \circledast \mathbf{h} h′=K⊛h
为了保留原来的heatmap的大小(?magnitude),最终会放缩一下
h ′ = h ′ − min ( h ′ ) max ( h ′ ) − min ( h ′ ) ∗ max ( h ) \boldsymbol{h}^{\prime}=\frac{\boldsymbol{h}^{\prime}-\min \left(\boldsymbol{h}^{\prime}\right)}{\max \left(\boldsymbol{h}^{\prime}\right)-\min \left(\boldsymbol{h}^{\prime}\right)} * \max (\boldsymbol{h}) h′=max(h′)−min(h′)h′−min(h′)∗max(h)
Coordinate Encoding
这里是讲热力图编码的方法
简单来说就是生成的时候不要用取整的方法得到热力图中心,而是直接生成