协方差矩阵(协方差矩阵的意义)

更新时间:2023-03-01 13:30:56 阅读: 评论:0

这篇文章的目标是从数据预处理的基础知识到深度学习中使用的现代技术。我的观点是我们可以使用代码(例如Python / NumPy)来更好地理解抽象的数学概念。

我们将从数据科学和机器学习/深度学习中的基本但非常有用的概念开始,如方差和协方差矩阵。我们将进一步介绍一些用于将图像输入神经网络的预处理技术。我们将尝试使用Python代码获得更具体的见解,以实际了解每个等式的作用。

预处理是指在将原始数据馈送到机器学习或深度学习算法之前对原始数据进行的所有转换。例如,在原始图像上训练卷积神经网络可能会导致糟糕的分类性能。预处理对于加速训练也很重要。

背景

A.方差和协方差

方差是在概率论和统计方差衡量随机变量或一组数据时离散程度的度量。协方差表示的是两个变量的总体的误差,这与只表示一个变量误差的方差不同。 如果两个变量的变化趋势一致,也就是说如果其中一个大于自身的期望值,另外一个也大于自身的期望值,那么两个变量之间的协方差就是正值。 如果两个变量的变化趋势相反,即其中一个大于自身的期望值,另外一个却小于自身的期望值,那么两个变量之间的协方差就是负值。

协方差值取决于变量的大小,因此很难对其进行分析。可以使用更容易解释的相关系数。相关系数就是归一化协方差。

上图中:正协方差意味着一个变量的大值与另一个变量的大值相关联(左)。负协方差意味着一个变量的大值与另一个变量的小值相关联(右)。

协方差矩阵是一个矩阵,它总结了一组向量的方差和协方差,它可以告诉你关于变量的很多事情。对角线对应于每个向量的方差:

矩阵 A及其协方差矩阵。对角线对应于每个列向量的方差

我们来看看方差的公式:

例如,A的第一列向量的方差是:

这是我们协方差矩阵的第一个单元格。对角线上的第二个元素对应于第二列向量与A的方差,依此类推。

注:从矩阵A中提取的向量对应于矩阵A的列。

其他单元格对应的是来自a的两个列向量之间的协方差,例如,第一列和第三列之间的协方差位于协方差矩阵中,即第1列和第3行(或第3列和第1行)。

让我们检查A的第一和第三列向量之间的协方差是否等于-2.67。两个变量X和Y之间的协方差公式为:

变量X和Y是上面例子中的第一和第三列向量。让我们分开这个公式,以确保它非常清晰:

1、求和符号(Σ)意味着我们将遍历元素的向量。我们从第一个元素(i=1)开始计算X的第一个元素减去向量X的均值。

2.将结果乘以Y的第一个元素减去向量Y的平均值。

3.重复向量的每个元素的过程并计算所有结果的总和。

4.除以向量中的元素数。

例1

让我们从矩阵A开始:

我们将计算第一个和第三个列向量之间的协方差:

x̄ = 3,ȳ= 4,n = 3,所以我们有:

就是协方差矩阵的值。

简单的方法。使用NumPy,可以使np.cov用函数计算协方差矩阵。

值得注意的是,如果您希望NumPy将列用作向量,则rowvar=Fal,必须使用该参数。此外,bias=True 除以n 而不是n-1。

让我们先用Python创建数组:

# First things first: let's import some librariesimport numpy as npimport matplotlib.pyplot as pltimport aborn as sns A = np.array([[1, 3, 5], [5, 4, 1], [3, 8, 6]])

现在我们将使用NumPy函数计算协方差:

np.cov(A, rowvar=Fal, bias=True)

array([[ 2.66666667, 0.66666667, -2.66666667],

[ 0.66666667, 4.66666667, 2.33333333],

[-2.66666667, 2.33333333, 4.66666667]])

使用点积查找协方差矩阵

还有另一种计算A的协方差矩阵的方法。你可以将A居中于0左右。从向量的每个元素中减去向量的平均值,得到一个平均值等于0的向量。它乘以自己的转置,并除以观察的数量。 让我们从一个Python实现开始:

def calculateCovariance(X): meanX = np.mean(X, axis = 0) lenX = X.shape[0] X = X - meanX covariance = X.T.dot(X)/lenXreturn covariance

让我们在矩阵A上测试它:

calculateCovariance(A)

array([[ 2.66666667, 0.66666667, -2.66666667],

[ 0.66666667, 4.66666667, 2.33333333],

[-2.66666667, 2.33333333, 4.66666667]])

我们最终得到与以前相同的结果。

解释很简单。两个向量之间的点积可以表示为:

它是向量的每个元素的乘积之和:

点积对应于矢量的每个元素的乘积之和

如果n是我们向量中的元素数,并且我们除以n:

你可以注意到,这与我们之前看到的协方差公式相差不远:

唯一的区别是,在协方差公式中,我们从每个元素中减去向量的均值。这就是为什么我们需要在做点积之前对数据进行center处理。

现在,如果我们有一个矩阵A,A和它的转置之间的点积将给你一个新的矩阵:

如果从zero-centered矩阵开始,此矩阵与其转置之间的点积将给出每个向量的方差和它们之间的协方差,即协方差矩阵。

B.可视化数据和协方差矩阵

为了获得关于协方差矩阵以及它如何有用的更多见解,我们将创建一个函数来将其与2D数据一起可视化。

该函数将如上所述计算协方差矩阵。它将创建两个子图 - 一个用于协方差矩阵,另一个用于数据。Seaborn的heatmap()函数用于创建颜色渐变 - 小值将以浅绿色着色,大值以深蓝色着色。我们选择了一种调色板颜色,但您可能更喜欢其他颜色。数据表示为散点图。

def plotDataAndCov(data): ACov = np.cov(data, rowvar=Fal, bias=True) print 'Covariance matrix: ', ACov fig, ax = plt.subplots(nrows=1, ncols=2) fig.t_size_inches(10, 10) ax0 = plt.subplot(2, 2, 1) # Choosing the colors cmap = sns.color_palette("GnBu", 10) sns.heatmap(ACov, cmap=cmap, vmin=0) ax1 = plt.subplot(2, 2, 2) # data can include the colors if data.shape[1]==3: c=data[:,2] el: c="#0A98BE" ax1.scatter(data[:,0], data[:,1], c=c, s=40) # Remove the top and right axes from the data plot ax1.spines['right'].t_visible(Fal)ax1.spines['top'].t_visible(Fal)

c.模拟数据

不相关的数据

现在我们有了绘图函数,我们将生成一些随机数据来可视化协方差矩阵能告诉我们什么。我们将从使用NumPy函数np.random.normal()从正态分布中提取的一些数据开始。

使用NumPy从正态分布中绘制样本

该函数需要平均值,标准偏差和分布的观察数作为输入。我们将创建300个观测值的两个随机变量,标准差为1.第一个平均值为1,第二个平均值为2.如果我们从正态分布中随机抽取两组300个观测值,则两个矢量将是不相关的。

np.random.ed(1234)a1 = np.random.normal(2, 1, 300)a2 = np.random.normal(1, 1, 300)A = np.array([a1, a2]).TA.shape

(300,2)

注意1:我们用.T转置数据,因为原始的shape是(2,300),我们想要观察到的数据的行数(shape(300,2))。

注意2:我们使用np.random.ed函数进行再现性。下次运行单元格时将使用相同的随机数。

让我们看看数据是怎样的:

A[:10,:]

array([[ 2.47143516, 1.52704645],

[ 0.80902431, 1.7111124 ],

[ 3.43270697, 0.78245452],

[ 1.6873481 , 3.63779121],

[ 1.27941127, -0.74213763],

[ 2.88716294, 0.90556519],

[ 2.85958841, 2.43118375],

[ 1.3634765 , 1.59275845],

[ 2.01569637, 1.1702969 ],

[-0.24268495, -0.75170595]])

现在,我们可以检查分布是否正常:

sns.distplot(A[:,0], color="#53BB04")sns.distplot(A[:,1], color="#0A98BE")plt.show()plt.clo()

我们可以看到分布具有相同的标准偏差但是具有不同的均值(1和2)。这正是我们所要求的。

现在我们可以用我们的函数绘制数据集及其协方差矩阵:

plotDataAndCov(A)plt.show()plt.clo()

Covariance matrix:

[[ 0.95171641 -0.0447816 ]

[-0.0447816 0.87959853]]

我们可以在散点图上看到两个维度是不相关的。注意,我们有一个平均值为1(y轴),另一个平均值为2(x轴)。

协方差矩阵表明每个变量的方差非常大(约1),列1和列2的协方差非常小(约0),因为我们保证了两个向量是独立的,所以这是一致的。反之则未必正确:协方差为0并不保证独立性

相关数据

现在,让我们通过从另一列中指定一列来构造依赖数据。Python代码如下:

np.random.ed(1234)b1 = np.random.normal(3, 1, 300)b2 = b1 + np.random.normal(7, 1, 300)/2.B = np.array([b1, b2]).TplotDataAndCov(B)plt.show()plt.clo()

Covariance matrix:

[[ 0.95171641 0.92932561]

[ 0.92932561 1.12683445]]

在散点图上可以看到两个维度之间的相关性。我们可以看到可以绘制一条线并用于从x预测y,反之亦然。协方差矩阵不是对角线的(对角线外有非零单元)。这意味着维度之间的协方差不为零。

2.预处理

A.平均归一化

平均归一化只是去除每次观察的平均值。

其中X'是归一化数据集,X是原始数据集,x̅是X的平均值。

均值归一化具有将数据居中于0的效果。我们将创建执行此操作的函数center():

def center(X): newX = X - np.mean(X, axis = 0) return newX

让我们试试我们之前创建的矩阵B:

BCentered = center(B) print 'Before: ' plotDataAndCov(B)plt.show()plt.clo() print 'After: ' plotDataAndCov(BCentered)plt.show()plt.clo()

Before:

Covariance matrix:

[[ 0.95171641 0.92932561]

[ 0.92932561 1.12683445]]

After:

Covariance matrix:

[[ 0.95171641 0.92932561]

[ 0.92932561 1.12683445]]

第一个图再次显示原始数据B,第二个图显示中心数据(look at the scale)。

B.标准化or归一化

标准化用于将所有特征放在相同的比例上。每个零中心尺寸除以其标准偏差。

其中X'是标准化的数据集,X是原始数据集,x̅ 是的平均值X, 并且σ是标准偏差X。

def standardize(X): newX = center(X)/np.std(X, axis = 0) return newX

让我们创建另一个具有不同scale 的数据集来检查它是否工作。

np.random.ed(1234)c1 = np.random.normal(3, 1, 300)c2 = c1 + np.random.normal(7, 5, 300)/2.C = np.array([c1, c2]).T plotDataAndCov(C)plt.xlim(0, 15)plt.ylim(0, 15)plt.show()plt.clo()

Covariance matrix:

[[ 0.95171641 0.83976242]

[ 0.83976242 6.22529922]]

我们可以看到x和y的尺度不同。另请注意,由于scale 差异,相关性似乎较小。现在让我们标准化它:

CStandardized = standardize(C) plotDataAndCov(CStandardized)plt.show()plt.clo()

Covariance matrix:

[[ 1. 0.34500274]

[ 0.34500274 1. ]]

看起来不错。您可以看到scales 相同,并且数据集根据两个轴以零为中心。

现在,看一下协方差矩阵。您可以看到每个坐标的方差 - 左上角单元格和右下角单元格 - 等于1。

这个新的协方差矩阵实际上是相关矩阵。两个变量(c1和c2)之间的Pearson相关系数是0.54220151。

C.白化

白化,或球化,数据意味着我们要转换它,使它有一个协方差矩阵它是单位矩阵对角线上是1,其他单元格是0。它被称为白噪声。

白化比其他预处理要复杂一些,但我们现在拥有了所需的所有工具。它涉及以下步骤:

Zero-center the data Decorrelate the data Rescale the data

让我们再次采取C并尝试执行这些步骤。

1.Zero-centering

这指的是均值归一化(2. A)。请查看有关该center()函数的详细信息。

CCentered = center(C) plotDataAndCov(CCentered)plt.show()plt.clo()

Covariance matrix:

[[ 0.95171641 0.83976242]

[ 0.83976242 6.22529922]]

2. Decorrelate

此时,我们需要对数据进行去相关。直觉上,这意味着我们想要旋转数据直到不再存在相关性。看下面的图片,

左边的图显示了相关的数据。例如,如果取一个数据点的x值很大,那么相关联的y也会很大。

现在取所有数据点,做一个旋转(大概是逆时针方向45度左右)。右边绘制的新数据不再相关。可以看到,大小y值与相同类型的x值相关。

问题是:我们如何找到正确的旋转来得到不相关的数据?

实际上,它正是协方差矩阵的特征向量所做的。它们表示数据扩散最大的方向:

协方差矩阵的特征向量为您提供最大化方差的方向。绿线的方向是方差最大的地方。只要看看这条线上预测的最小和最大点 - 差距很大。将其与橙色线上的projection相比- spread 非常小。

因此,我们可以通过使用特征向量projection 来解相关数据。这将具有应用所需旋转并移除尺寸之间的相关性的效果。以下是步骤:

计算协方差矩阵计算协方差矩阵的特征向量将特征向量矩阵应用于数据 - 这将应用旋转

让我们将其打包到函数中:

def decorrelate(X): newX = center(X) cov = X.T.dot(X)/float(X.shape[0]) # Calculate the eigenvalues and eigenvectors of the covariance matrix eigVals, eigVecs = np.linalg.eig(cov) # Apply the eigenvectors to X decorrelated = X.dot(eigVecs) return decorrelated

让我们试着去掉我们的zero-centered的矩阵C来看它的实际效果:

plotDataAndCov(C)plt.show()plt.clo() CDecorrelated = decorrelate(CCentered)plotDataAndCov(CDecorrelated)plt.xlim(-5,5)plt.ylim(-5,5)plt.show()plt.clo()

Covariance matrix:

[[ 0.95171641 0.83976242]

[ 0.83976242 6.22529922]]

Covariance matrix:

[[ 5.96126981e-01 -1.48029737e-16]

[ -1.48029737e-16 3.15205774e+00]]

这很有效。

我们可以看到相关性不再存在了。协方差矩阵,现在是对角矩阵,证实了两个维度之间的协方差等于0。

3.Rescale the data

下一步是缩放不相关矩阵以获得对应于单位矩阵的协方差矩阵。为此,我们通过将每个维度除以其对应特征值的平方根来缩放我们的去相关数据。

def whiten(X): newX = center(X) cov = X.T.dot(X)/float(X.shape[0]) # Calculate the eigenvalues and eigenvectors of the covariance matrix eigVals, eigVecs = np.linalg.eig(cov) # Apply the eigenvectors to X decorrelated = X.dot(eigVecs) # Rescale the decorrelated data whitened = decorrelated / np.sqrt(eigVals + 1e-5)return whitened

注意:我们添加一个小值(此处为10 ^ -5)以避免除以0。

CWhitened = whiten(CCentered) plotDataAndCov(CWhitened)plt.xlim(-5,5)plt.ylim(-5,5)plt.show()plt.clo()

Covariance matrix:

[[ 9.99983225e-01 -1.06581410e-16]

[ -1.06581410e-16 9.99996827e-01]]

我们可以看到,使用协方差矩阵,这一切都很好。我们的东西看起来像一个单位矩阵 - 对角线上1,其他地方0。

3.图像白化

我们将看到如何应用白化来预处理图像数据集。为此,我们将使用Pal&Sudeep(2016)的论文(https://ieeexplore.ieee.org/document/7808140/),其中提供了有关该过程的一些细节。这种预处理技术称为零分量分析(ZCA)。

查看论文,但这是他们得到的结果。显示原始图像(左)和ZCA(右)后的图像。

从CIFAR10数据集中白化图像。Pal&Sudeep(2016)的论文结果 。

首先要做的事情。我们将从CIFAR数据集加载图像。此数据集可从Keras获得,您也可以在此处下载(https://www.cs.toronto.edu/~kriz/cifar.html)。

from keras.datats import cifar10 (X_train, y_train), (X_test, y_test) = cifar10.load_data() X_train.shape

(50000, 32, 32, 3)

CIFAR10数据集的训练集包含50000个图像。X_train是(50000, 32, 32, 3)。每幅图像为32像素×32像素,每个像素包含3个维度(R,G,B)。每个值是0到255之间相应颜色的亮度。

我们将从仅选择图像的一个子集开始,假设为1000:

X = X_train[:1000]print X.shape

(1000,32,32,3)

现在我们将对阵列进行reshape ,以获得每行一个图像的平面图像数据。每个图像都是(1, 3072)因为32 x 32 x 3 = 3072.因此,包含所有图像的数组将是(1000, 3072):

X = X.reshape(X.shape[0], X.shape[1]*X.shape[2]*X.shape[3])print X.shape

(1000,3072)

下一步是能够看到图像。Matplotlib中的函数imshow()可用于显示图像。它需要具有shape的图像(M x N x 3),所以让我们创建一个函数来reshape图像并能够从shape(1,3072)中可视化它们。

def plotImage(X): plt.figure(figsize=(1.5, 1.5)) plt.imshow(X.reshape(32,32,3)) plt.show() plt.clo()

例如,让我们绘制我们加载的图像之一:

plotImage(X[12, :])

我们现在可以实现图像的白化。Pal&Sudeep(2016)描述了这个过程:

1.第一步是通过除以255(像素的最大值)来重新缩放图像以获得范围[0,1]。

回想一下,获得范围[0,1]的公式是:

但是,这里,最小值为0,因此这导致:

X_norm = X / 255.print 'X.min()', X_norm.min()print 'X.max()', X_norm.max()

X.min() 0.0

X.max() 1.0

2.从所有图像中减去均值。

一种方法是拍摄每张图像并从每个像素中删除该图像的平均值(Jarrett等,2009)。这个过程背后的直觉是它将每个图像的像素集中在0左右。

另一种方法是为每个图像拍摄我们拥有的3072个像素(RGB,32乘32像素),并在所有图像中减去该像素的平均值。根据所有图像,每个像素将以0为中心。当您使用图像为网络提供信息时,每个像素都被视为不同的特征。通过每像素平均减法,我们将每个特征(像素)集中在0左右。这种技术是常用的。

我们现在将从1000张图像中进行每像素平均减法。我们的数据按这些维度进行组织(images, pixels)。这是(1000, 3072)因为有1000张图像,32 x 32 x 3 = 3072像素。因此,平均每像素可以从第一轴得到:

X_norm.mean(axis = 0).shape

(3072,)

这就得到3072个值,也就是均值的数量每像素1。

X_norm.mean(axis=0)

array([ 0.5234 , 0.54323137, 0.5274 , …, 0.50369804,

0.50011765, 0.45227451])

这接近0.5,因为我们已经归一化到范围[0,1]。但是,我们仍然需要从每个像素中删除均值:

X_norm = X_norm - X_norm.mean(axis=0)

为了说服自己它有效,我们将计算第一个像素的平均值。我们希望它是0。

X_norm.mean(axis=0)

array([ -5.30575583e-16, -5.98021632e-16, -4.23439062e-16, …,

-1.81965554e-16, -2.49800181e-16, 3.98570066e-17])

这不完全是0,但它足够小,我们可以认为它有效!

现在我们想要计算零中心数据的协方差矩阵。就像我们在上面看到的那样,我们可以使用NumPy中的np.cov()函数来计算它。

请注意,我们的变量是不同的图像。这意味着变量是矩阵x的行。为了明确一点,我们将用参数rowvar=TRUE将这个信息告诉NumPy,即使它默认为TRUE:

cov = np.cov(X_norm, rowvar=True)

现在是神奇的部分 - 我们将计算协方差矩阵的奇异值和向量,并使用它们来旋转我们的数据集。

注意:大量图像可能需要一些时间,这就是我们仅使用1000的原因。在论文中,他们使用了10000张图像。您可以根据您使用的图像数量来比较结果:

U,S,V = np.linalg.svd(cov)

在论文中,他们使用以下等式:

其中U为左奇异向量,S为图像初始归一化数据集协方差的奇异值,X为归一化数据集。ϵhyper-parameter叫做白化系数。diag(a)对应于一个矩阵,向量a是对角线,其他单元格中为0。

我们将尝试实现这个方程。让我们从检查SVD的维数开始:

print U.shape, S.shape

(1000, 1000) (1000,)

S是一个包含1000个元素(奇异值)的向量。因此,diag(S)的shape为(1000,1000),以S为对角线:

print np.diag(S)print ' shape:', np.diag(S).shape

[[ 8.15846654e+00 0.00000000e+00 0.00000000e+00 …, 0.00000000e+00

0.00000000e+00 0.00000000e+00]

[ 0.00000000e+00 4.68234845e+00 0.00000000e+00 …, 0.00000000e+00

0.00000000e+00 0.00000000e+00]

[ 0.00000000e+00 0.00000000e+00 2.41075267e+00 …, 0.00000000e+00

0.00000000e+00 0.00000000e+00]

…,

[ 0.00000000e+00 0.00000000e+00 0.00000000e+00 …, 3.92727365e-05

0.00000000e+00 0.00000000e+00]

[ 0.00000000e+00 0.00000000e+00 0.00000000e+00 …, 0.00000000e+00

3.52614473e-05 0.00000000e+00]

[ 0.00000000e+00 0.00000000e+00 0.00000000e+00 …, 0.00000000e+00

0.00000000e+00 1.35907202e-15]]

shape: (1000, 1000)

检查这部分:

shape(1000、1000) 。我们也看到X的shape是(1000 3072)X_ZCA的形状为:

它对应于初始数据集的shape。

epsilon = 0.1X_ZCA = U.dot(np.diag(1.0/np.sqrt(S + epsilon))).dot(U.T).dot(X_norm) plotImage(X[12, :])plotImage(X_ZCA[12, :])

令人失望!如果你看一下这篇论文,这不是他们所展示的那种结果。实际上,这是因为我们没有重新调整像素,并且存在负值。为此,我们可以使用与上述相同的技术将其放回[0,1]范围内:

X_ZCA_rescaled = (X_ZCA - X_ZCA.min()) / (X_ZCA.max() - X_ZCA.min())print 'min:', X_ZCA_rescaled.min()print 'max:', X_ZCA_rescaled.max()

min: 0.0

max: 1.0

plotImage(X[12, :])plotImage(X_ZCA_rescaled[12, :])

很棒!它看起来像论文上的图像。如前所述,他们使用了10000张图像而不是像我们这样的1000张。

要根据您使用的图像数量和超参数ε的效果来查看结果的差异,以下是不同值的结果:

根据我们使用的图像的数量和超参数ε的值,白化的结果是不同的。左侧的图像是原始图像。在论文中,Pal&Sudeep(2016)使用了10000个图像,epsilon = 0.1。这对应于左下图像。

本文发布于:2023-02-28 20:00:00,感谢您对本站的认可!

本文链接:https://www.wtabcd.cn/zhishi/a/167764865673741.html

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。

本文word下载地址:协方差矩阵(协方差矩阵的意义).doc

本文 PDF 下载地址:协方差矩阵(协方差矩阵的意义).pdf

标签:协方差   矩阵   意义
相关文章
留言与评论(共有 0 条评论)
   
验证码:
Copyright ©2019-2022 Comsenz Inc.Powered by © 实用文体写作网旗下知识大全大全栏目是一个全百科类宝库! 优秀范文|法律文书|专利查询|