通过矩阵求两个向量集中元素两两之间的欧氏距离(python实现)

在很多算法中都会涉及到求向量欧式距离,例如机器学习中的KNN算法,就需要对由训练集A和测试集B中的向量组成的所有有序对 (Ai,Bi) ,求出 Ai Bi 的欧式距离。这样的话就会带来一个二重的嵌套循环,在向量集很大时效率不高。

这里介绍如何将这一过程用矩阵运算实现。

假设有两个三维向量集,用矩阵表示:

A=[a11a12a21a22a31a32]

B=b11b12b13b21b22b23b31b32b33

要求A,B两个集合中的元素两两间欧氏距离。

先求出 ABT

ABT=k=13ak1bk1k=13ak2bk1k=13ak1bk2k=13ak2bk2k=13ak1bk3k=13ak2bk3

然后对 A BT 分别求其中每个向量的模平方,并扩展为2*3矩阵:

Asq=k=13(ak1)2k=13(ak2)2k=13(ak1)2k=13(ak2)2k=13(ak1)2k=13(ak2)2

Bsq=k=13(bk1)2k=13(bk1)2k=13(bk2)2k=13(bk2)2k=13(bk3)2k=13(bk3)2

然后:

Asq+Bsq2ABT=k=13(ak1bk1)2k=13(ak2bk1)2k=13(ak1bk2)2k=13(ak2bk2)2k=13(ak1bk3)2k=13(ak2bk3)2

将上面这个矩阵一开平方,就得到了A,B向量集两两间的欧式距离了。

下面是python实现:

import numpy
def EuclideanDistances(A, B):
    BT = B.transpose()
    vecProd = A * BT
    SqA =  A.getA()**2
    sumSqA = numpy.matrix(numpy.sum(SqA, axis=1))
    sumSqAEx = numpy.tile(sumSqA.transpose(), (1, vecProd.shape[1]))    
    SqB = B.getA()**2
    sumSqB = numpy.sum(SqB, axis=1)
    sumSqBEx = numpy.tile(sumSqB, (vecProd.shape[0], 1))    
    SqED = sumSqBEx + sumSqAEx - 2*vecProd   
    ED = (SqED.getA())**0.5
    return numpy.matrix(ED)
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
欧氏距离的公式是:d(x,y) = sqrt((x1-y1)^2 + (x2-y2)^2 + ... + (xn-yn)^2) 以下是使用CUDA实现的代码示例: ```C++ #include <stdio.h> #include <stdlib.h> #include <cuda_runtime.h> #define N 1024 __global__ void euclideanDistance(float *x, float *y, float *result) { int tid = blockIdx.x * blockDim.x + threadIdx.x; if (tid < N) { float sum = 0.0f; for (int i = 0; i < N; i++) { float diff = x[i] - y[i]; sum += diff * diff; } result[tid] = sqrt(sum); } } int main() { float *x, *y, *result; cudaMallocManaged(&x, N * sizeof(float)); cudaMallocManaged(&y, N * sizeof(float)); cudaMallocManaged(&result, N * sizeof(float)); // initialize x and y with random values for (int i = 0; i < N; i++) { x[i] = static_cast<float>(rand()) / RAND_MAX; y[i] = static_cast<float>(rand()) / RAND_MAX; } int blockSize = 256; int numBlocks = (N + blockSize - 1) / blockSize; euclideanDistance<<<numBlocks, blockSize>>>(x, y, result); cudaDeviceSynchronize(); // print the result for (int i = 0; i < N; i++) { printf("Distance between x and y[%d] = %f\n", i, result[i]); } cudaFree(x); cudaFree(y); cudaFree(result); return 0; } ``` 在这个示例中,我们使用了CUDA的并行计算能力,通过在GPU上同时计算多个距离来加速计算。首先,我们在GPU上分配了内存用于存储两个向量和计算结果。然后,我们使用随机数初始化了两个向量。接下来,我们将距离计算函数euclideanDistance定义为一个CUDA核函数,在每个线程中计算两个向量之间的距离。最后,我们在主程序中调用这个核函数,并用cudaDeviceSynchronize()等待所有线程完成计算。最后打印结果并释放内存。 请注意,由于这个示例使用了CUDA,因此需要在支持CUDA的GPU上运行。如果您的计算机不支持CUDA,或者您没有安装CUDA,那么您将无法运行这个示例。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值