在处理一组数据时,通常想做的第一件事就是了解变量的分布情况。这一章将简要介绍 seaborn
中用于检查单变量和双变量分布的一些工具。
import numpy as np
import pandas as pd
import seaborn as sns
import matplotlib.pyplot as plt
from scipy import stats
import warnings
warnings.filterwarnings('ignore')
sns.set(color_codes=True)
在seaborn
中想要快速查看单变量分布的最方便的方法是使用distplot()
函数。默认情况下,该方法将会绘制直方图histogram
并拟合[内核密度估计] kernel density estimate (KDE)
.
x = np.random.normal(size=100)
sns.distplot(x)
<Axes: ylabel='Density'>
对于直方图我们可能很熟悉,而且matplotlib
中已经存在hist
函数。 直方图首先确定数据区间,然后观察数据落入这些区间中的数量来绘制柱形图以此来表征数据的分布情况。 为了说明这一点,让我们删除密度曲线并添加一个rug plot
,它在每个观察值上画一个小的垂直刻度。您可以使用rugplot()
函数来创建rugplot
本身,但是也可以在distplot()
中使用:
sns.distplot(x, kde=False, rug=True)
<Axes: >
在绘制柱状图时,您的主要选择是要使用的“桶”的数量和放置它们的位置。distplot()
使用一个简单的规则来很好地猜测默认情况下正确的数字是多少,但是尝试更多或更少的“桶”可能会揭示数据中的其他特性:
sns.distplot(x, bins=20, kde=False, rug=True)
<Axes: >
可能你对核密度估计不太熟悉,但它可以是绘制分布形状的有力工具。和直方图一样,KDE 图沿另一个轴的高度,编码一个轴上的观测密度:
sns.distplot(x, hist=False, rug=True)
<Axes: ylabel='Density'>
我们可以看到,如果在seaborn
中使用kdeplot()
函数, 我们可以得到相同的曲线。这个函数也被distplot()
所使用, 但是当您只想要密度估计时,它提供了一个更直接的接口,可以更容易地访问其他选项:
sns.kdeplot(x, shade=True)
<Axes: ylabel='Density'>
KDE
的带宽(bw
)参数控制估计与数据的拟合程度,就像直方图中的bin
大小一样。 它对应于我们在上面绘制的内核的宽度。 默认行为尝试使用常用参考规则猜测一个好的值,但尝试更大或更小的值可能会有所帮助:
sns.kdeplot(x)
sns.kdeplot(x, bw=.2, label="bw: 0.2")
sns.kdeplot(x, bw=2, label="bw: 2")
plt.legend()
<matplotlib.legend.Legend at 0x12ac30350>
正如您在上面所看到的,高斯KDE
过程的本质意味着估计超出了数据集中最大和最小的值。有可能控制超过极值多远的曲线是由cut
参数绘制的;然而,这只影响曲线的绘制方式,而不影响曲线的拟合方式:
sns.kdeplot(x, shade=True, cut=0)
sns.rugplot(x)
<Axes: ylabel='Density'>
还可以使用distplot()
将参数分布拟合到数据集上,并直观地评估其与观测数据的对应程度:
x = np.random.gamma(6, size=200)
sns.distplot(x, kde=False, fit=stats.gamma)
<Axes: >
它对于可视化两个变量的二元分布也很有用。在seaborn
中,最简单的方法就是使用jointplot()
函数,它创建了一个多面板图形,显示了两个变量之间的二元(或联合)关系,以及每个变量在单独轴上的一元(或边际)分布。
mean, cov = [0, 1], [(1, .5), (.5, 1)]
data = np.random.multivariate_normal(mean, cov, 200)
df = pd.DataFrame(data, columns=["x", "y"])
可视化二元分布最常见的方法是散点图,其中每个观察点都以x
和y
值表示。 这类似于二维rug plot
。 您可以使用matplotlib
的plt.scatter
函数绘制散点图, 它也是jointplot()
函数显示的默认类型的图:
sns.jointplot(x="x", y="y", data=df)
<seaborn.axisgrid.JointGrid at 0x12ae8b350>
类似于单变量的直方图,用于描绘二元变量关系的图称为 “hexbin
” 图,因为它显示了落入六边形“桶”内的观察计数。 此图对于相对较大的数据集最有效。它可以通过调用 matplotlib
中的plt.hexbin
函数获得并且在jointplot()
作为一种样式。当使用白色作为背景色时效果最佳。
x, y = np.random.multivariate_normal(mean, cov, 1000).T
with sns.axes_style("white"):
sns.jointplot(x=x, y=y, kind="hex", color="k")
也可以使用上面描述的核密度估计过程来可视化二元分布。在seaborn
中,这种图用等高线图表示, 在jointplot()
中被当作一种样式:
sns.jointplot(x="x", y="y", data=df, kind="kde")
<seaborn.axisgrid.JointGrid at 0x12b3a6ad0>
如果希望更连续地显示双变量密度,可以简单地增加轮廓层的数量:
f, ax = plt.subplots(figsize=(6, 6))
cmap = sns.cubehelix_palette(as_cmap=True, dark=0, light=1, reverse=True)
sns.kdeplot(x=df.x, y=df.y, cmap=cmap, n_levels=60, shade=True)
<Axes: xlabel='x', ylabel='y'>
jointplot()
函数使用JointGrid
来管理图形。为了获得更大的灵活性,您可能想直接使用JointGrid
来绘制图形。jointplot()
在绘图后返回JointGrid
对象,您可以使用它添加更多图层或调整可视化的其他方面:
g = sns.jointplot(x="x", y="y", data=df, kind="kde", color="m")
g.plot_joint(plt.scatter, c="w", s=30, linewidth=1, marker="+")
g.ax_joint.collections[0].set_alpha(0)
g.set_axis_labels("$X$", "$Y$")
<seaborn.axisgrid.JointGrid at 0x12b729b90>
要在数据集中绘制多个成对的双变量分布,您可以使用pairplot()
函数。 这将创建一个轴矩阵并显示DataFrame
中每对列的关系,默认情况下,它还绘制对角轴上每个变量的单变量分布:
iris = sns.load_dataset("iris")
sns.pairplot(iris)
<seaborn.axisgrid.PairGrid at 0x12b6f8710>
与jointplot()
和JointGrid
之间的关系非常类似, pairplot()
函数构建在PairGrid
对象之上, 可以直接使用它来获得更大的灵活性:
g = sns.PairGrid(iris)
g.map_diag(sns.kdeplot)
g.map_offdiag(sns.kdeplot, n_levels=6)
<seaborn.axisgrid.PairGrid at 0x12bdfd510>