1

我知道scipy.spatial.distance.pdist函数以及如何从结果矩阵/ndarray 计算平均值。

>>> x = np.random.rand(10000, 2)
>>> y = pdist(x, metric='euclidean')
>>> y.mean()
0.5214255824176626

在上面的示例中y变得非常大(几乎是输入数组的 2,500 倍):

>>> y.shape
(49995000,)
>>> from sys import getsizeof
>>> getsizeof(x)
160112
>>> getsizeof(y)
399960096
>>> getsizeof(y) / getsizeof(x)
2498.0019986009793

但由于我只对平均成对距离感兴趣,因此不必将距离矩阵保存在内存中。相反,可以单独计算每行(或列)的平均值。然后可以从行平均值计算最终平均值。

是否已经存在利用此属性的功能,或者是否有一种简单的方法可以扩展/组合现有功能来做到这一点?

4

2 回答 2

1

如果使用距离的平方版本,则相当于使用 n-1 的方差:

from scipy.spatial.distance import pdist, squareform
import numpy as np
x = np.random.rand(10000, 2)
y = np.array([[1,1], [0,0], [2,0]])
print(pdist(x, 'sqeuclidean').mean())
print(np.var(x, 0, ddof=1).sum()*2)
>>0.331474285845873
0.33147428584587346
于 2019-05-02T19:56:59.453 回答
0

您必须根据构成平均值的观察数对每一行进行加权。例如,3 x 2 矩阵的 pdist 是方形 3 x 3 距离矩阵的扁平上三角形(偏移量为 1)。

arr = np.arange(6).reshape(3,2)
arr
array([[0, 1],
       [2, 3],
       [4, 5]])
pdist(arr)
array([2.82842712, 5.65685425, 2.82842712])
from sklearn.metrics import pairwise_distances
square = pairwise_distances(arr)
square
array([[0.        , 2.82842712, 5.65685425],
       [2.82842712, 0.        , 2.82842712],
       [5.65685425, 2.82842712, 0.        ]])
square[triu_indices(square.shape[0], 1)]
array([2.82842712, 5.65685425, 2.82842712])

有一个pairwise_distances_chuncked函数可用于逐行迭代距离矩阵,但您需要跟踪行索引以确保仅取矩阵上/下三角形中的值的平均值(距离矩阵是对称的)。这并不复杂,但我想你会引入显着的放缓。

tot = ((arr.shape[0]**2) - arr.shape[0]) / 2
weighted_means = 0
for i in gen:
    if r < arr.shape[0]:
        sm = i[0, r:].mean()
        wgt = (i.shape[1] - r) / tot
        weighted_means += sm * wgt
       r += 1
于 2019-05-02T20:10:30.900 回答