scipy.special.pseudo_huber#
- scipy.special.pseudo_huber(delta, r, out=None) = <ufunc 'pseudo_huber'>#
伪 Huber 损失函数。
\[\mathrm{pseudo\_huber}(\delta, r) = \delta^2 \left( \sqrt{ 1 + \left( \frac{r}{\delta} \right)^2 } - 1 \right)\]- 参数:
- deltaarray_like
输入数组,表示软二次损失与线性损失的改变点。
- rarray_like
输入数组,可能表示残差。
- outndarray, optional
函数结果的可选输出数组
- 返回:
- res标量或 ndarray
计算出的伪 Huber 损失函数值。
另请参阅
huber
与此函数相似的近似函数
说明
与
huber
类似,pseudo_huber
通常用作统计学或机器学习中鲁棒的损失函数,以减少异常值的影响。与huber
不同,pseudo_huber
是平滑的。通常,r 表示残差,即模型预测与数据之间的差异。然后,对于 \(|r|\leq\delta\),
pseudo_huber
类似于平方误差,对于 \(|r|>\delta\),则类似于绝对误差。这样,伪 Huber 损失通常在模型拟合中对小残差(如平方误差损失函数)实现快速收敛,并且仍然像绝对误差损失一样减少异常值 (\(|r|>\delta\)) 的影响。由于 \(\delta\) 是平方误差和绝对误差机制之间的截止点,因此必须针对每个问题仔细调整。pseudo_huber
也是凸的,使其适用于基于梯度的优化。[1] [2]在 0.15.0 版本中添加。
参考文献
[1]Hartley, Zisserman,“计算机视觉中的多视图几何”。2003 年。剑桥大学出版社。第 619 页
[2]Charbonnier 等人。“计算成像中确定性的边缘保留正则化”。1997 年。IEEE Trans. Image Processing。6 (2): 298 - 311。
示例
导入所有必要的模块。
>>> import numpy as np >>> from scipy.special import pseudo_huber, huber >>> import matplotlib.pyplot as plt
计算
delta=1
在r=2
时的函数值。>>> pseudo_huber(1., 2.) 1.2360679774997898
通过为 delta 提供列表或 NumPy 数组,计算
r=2
在不同 delta 时的函数值。>>> pseudo_huber([1., 2., 4.], 3.) array([2.16227766, 3.21110255, 4. ])
通过为 r 提供列表或 NumPy 数组,计算
delta=1
在几个点上的函数值。>>> pseudo_huber(2., np.array([1., 1.5, 3., 4.])) array([0.47213595, 1. , 3.21110255, 4.94427191])
通过为具有兼容广播形状的 delta 和 r 提供数组,可以计算不同 delta 和 r 的函数值。
>>> r = np.array([1., 2.5, 8., 10.]) >>> deltas = np.array([[1.], [5.], [9.]]) >>> print(r.shape, deltas.shape) (4,) (3, 1)
>>> pseudo_huber(deltas, r) array([[ 0.41421356, 1.6925824 , 7.06225775, 9.04987562], [ 0.49509757, 2.95084972, 22.16990566, 30.90169944], [ 0.49846624, 3.06693762, 27.37435121, 40.08261642]])
绘制不同 delta 的函数。
>>> x = np.linspace(-4, 4, 500) >>> deltas = [1, 2, 3] >>> linestyles = ["dashed", "dotted", "dashdot"] >>> fig, ax = plt.subplots() >>> combined_plot_parameters = list(zip(deltas, linestyles)) >>> for delta, style in combined_plot_parameters: ... ax.plot(x, pseudo_huber(delta, x), label=rf"$\delta={delta}$", ... ls=style) >>> ax.legend(loc="upper center") >>> ax.set_xlabel("$x$") >>> ax.set_title(r"Pseudo-Huber loss function $h_{\delta}(x)$") >>> ax.set_xlim(-4, 4) >>> ax.set_ylim(0, 8) >>> plt.show()
最后,通过绘制
huber
和pseudo_huber
以及它们关于 r 的梯度,来说明它们之间的差异。该图显示,pseudo_huber
是连续可微的,而huber
在 \(\pm\delta\) 点不可微。>>> def huber_grad(delta, x): ... grad = np.copy(x) ... linear_area = np.argwhere(np.abs(x) > delta) ... grad[linear_area]=delta*np.sign(x[linear_area]) ... return grad >>> def pseudo_huber_grad(delta, x): ... return x* (1+(x/delta)**2)**(-0.5) >>> x=np.linspace(-3, 3, 500) >>> delta = 1. >>> fig, ax = plt.subplots(figsize=(7, 7)) >>> ax.plot(x, huber(delta, x), label="Huber", ls="dashed") >>> ax.plot(x, huber_grad(delta, x), label="Huber Gradient", ls="dashdot") >>> ax.plot(x, pseudo_huber(delta, x), label="Pseudo-Huber", ls="dotted") >>> ax.plot(x, pseudo_huber_grad(delta, x), label="Pseudo-Huber Gradient", ... ls="solid") >>> ax.legend(loc="upper center") >>> plt.show()