Вычислить попарно возведенную в квадрат разность нескольких векторов - PullRequest
0 голосов
/ 24 октября 2018

Я ищу самый быстрый способ вычисления квадрата разности между двумя векторами ((x1-x2)**2), но попарно (все комбинации или только верхний треугольник).

x1 = [1,3,5,6,8]
x2 = [3,6,7,9,12]

Ожидаемый результат:

array([[   4.,   25.,   36.,   64.,  121.],
       [   0.,    9.,   16.,   36.,   81.],
       [   4.,    1.,    4.,   16.,   49.],
       [   9.,    0.,    1.,    9.,   36.],
       [  25.,    4.,    1.,    1.,   16.]])

или

array([[   4.,   25.,   36.,   64.,  121.],
       [   0.,    9.,   16.,   36.,   81.],
       [   0.,    0.,    4.,   16.,   49.],
       [   0.,    0.,    0.,    9.,   36.],
       [   0.,    0.,    0.,    0.,    16.]])

или даже (если быстрее):

array([   4.,   25.,   36.,   64.,  121.,    9.,   16.,   36.,   81.,
      4.,    1.,    4.,   16.,   49.,    9.,    1.,    9.,   36.,
     25.,    4.,    1.,    1.,   16.])

Ответы [ 2 ]

0 голосов
/ 24 октября 2018

Вот один с broadcasting и masking, чтобы получить верхние треугольные, а затем возвести в квадрат только те, которые для повышения эффективности производительности -

def pairwise_squared_diff(x1, x2):
    x1 = np.asarray(x1)
    x2 = np.asarray(x2)
    diffs = x1[:,None] - x2
    mask = np.arange(len(x1))[:,None] <= np.arange(len(x2))
    return (diffs[mask])**2

Пробный прогон -

In [85]: x1
Out[85]: array([1, 3, 5, 6, 8])

In [86]: x2
Out[86]: array([ 3,  6,  7,  9, 12])

In [87]: pairwise_squared_diff(x1, x2)
Out[87]: 
array([  4,  25,  36,  64, 121,   9,  16,  36,  81,   4,  16,  49,   9,
        36,  16])

Возможные улучшения

Улучшение № 1:

Мы также можем использовать np.tri для генерации mask -

mask = ~np.tri(len(x1),len(x2),dtype=bool,k=-1)

Улучшение # 2:

Если у нас все в порядке с 2D выходом с нижними треугольными, установленными как 0s, то простое поэлементное умножение с mask решает егослишком, чтобы получить окончательный вывод -

(diffs*mask)**2

Это будет хорошо работать с numexpr модулем для больших данных и для повышения эффективности памяти и, следовательно, производительности.

Улучшение № 3:

Мы могли бы также вычислить различия с помощью numexpr и, следовательно, вычислить замаскированный вывод тоже с помощью того же метода evaulate, чтобы дать нам новое решение в целом -

def pairwise_squared_diff_numexpr(x1, x2):
    x1 = np.asarray(x1)
    x2 = np.asarray(x2)
    mask = ~np.tri(len(x1),len(x2),dtype=bool,k=-1)
    return ne.evaluate('mask*((x1D-x2)**2)',{'x1D':x1[:,None]})

Сроки с улучшениями

Давайте изучим этипредложения по производительности для больших массивов -

Настройка:

In [136]: x1 = np.random.randint(0,9,(1000))

In [137]: x2 = np.random.randint(0,9,(1000))

С улучшением # 1:

In [138]: %timeit np.arange(len(x1))[:,None] <= np.arange(len(x2))
1000 loops, best of 3: 772 µs per loop

In [139]: %timeit ~np.tri(len(x1),len(x2),dtype=bool,k=-1)
1000 loops, best of 3: 243 µs per loop

С улучшением # 2:

In [140]: import numexpr as ne

In [141]: diffs = x1[:,None] - x2
     ...: mask = np.arange(len(x1))[:,None] <= np.arange(len(x2))

In [142]: %timeit (diffs[mask])**2
1000 loops, best of 3: 1.46 ms per loop

In [143]: %timeit ne.evaluate('(diffs*mask)**2')
1000 loops, best of 3: 1.05 ms per loop

С улучшением № 3 для полных решений:

In [170]: %timeit pairwise_squared_diff(x1, x2)
100 loops, best of 3: 3.66 ms per loop

In [171]: %timeit pairwise_squared_diff_numexpr(x1, x2)
1000 loops, best of 3: 1.54 ms per loop

Loopy one

Для полноты, вот петлевый, который использует slicing, чтобы работать лучше, чем чистый broadcasting, благодаряк эффективности памяти -

def pairwise_squared_diff_loopy(x1,x2):
    n = len(x2)
    idx = np.concatenate(( [0], np.arange(n,0,-1).cumsum() ))
    start, stop = idx[:-1], idx[1:]
    L = n*(n+1)//2
    out = np.empty(L,dtype=np.result_type(x1,x2))
    for i,(s0,s1) in enumerate(zip(start,stop)):
        out[s0:s1] = x1[i] - x2[i:]
    return out**2

Сроки -

In [300]: x1 = np.random.randint(0,9,(1000))
     ...: x2 = np.random.randint(0,9,(1000))

In [301]: %timeit pairwise_squared_diff(x1, x2)
100 loops, best of 3: 3.44 ms per loop

In [302]: %timeit pairwise_squared_diff_loopy(x1, x2)
100 loops, best of 3: 2.73 ms per loop
0 голосов
/ 24 октября 2018

Вы можете использовать широковещательную рассылку:

x1 = np.asarray([1,3,5,6,8]).reshape(-1, 1)
x2 = np.asarray([3,6,7,9,12]).reshape(1, -1)
(x1 - x2)**2

Вывод:

array([[  4,  25,  36,  64, 121],
       [  0,   9,  16,  36,  81],
       [  4,   1,   4,  16,  49],
       [  9,   0,   1,   9,  36],
       [ 25,   4,   1,   1,  16]])

, который прост в кодировании, но вычисляет все значения, поэтому он может быть оптимизирован для вычисления только верхнего треугольника.

...