机器学习算法入门之(一)梯度下降法实现线性回归

1. 背景

文章的背景取自An Introduction to Gradient Descent and Linear Regression,本文想在该文章的基础上,完整地描述线性回归算法。部分数据和图片取自该文章。没有太多时间抠细节,所以难免有什么缺漏错误之处,望指正。

线性回归的目标很简单,就是用一条线,来拟合这些点,并且使得点集与拟合函数间的误差最小。如果这个函数曲线是一条直线,那就被称为线性回归,如果曲线是一条二次曲线,就被称为二次回归。数据来自于GradientDescentExample中的data.csv文件,共100个数据点,如下图所示:

图片名称

我们的目标是用一条直线来拟合这些点。既然是二维,那么 y=b+mx 这个公式相信对于中国学生都很熟悉。其中 b 是直线在y轴的截距(y-intercept), m 是直线的斜率(slope)。寻找最佳拟合直线的过程,其实就是寻找最佳的 b m 的过程。为了寻找最佳的拟合直线,这里首先要定义,什么样的直线才是最佳的直线。我们定义误差(cost function):

 Error(b,m)=1N1N((b+mxi)yi)2

计算损失函数的python代码如下:

<code class="hljs python has-numbering" style="display: block; padding: 0px; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-radius: 0px; word-wrap: normal; background: transparent;"><span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;"># y = b + mx</span>
<span class="hljs-function" style="box-sizing: border-box;"><span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">def</span> <span class="hljs-title" style="box-sizing: border-box;">compute_error_for_line_given_points</span><span class="hljs-params" style="color: rgb(102, 0, 102); box-sizing: border-box;">(b, m, points)</span>:</span>
    totalError = sum((((b + m * point[<span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">0</span>]) - point[<span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">1</span>]) ** <span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">2</span> <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">for</span> point <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">in</span> points))
    <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">return</span> totalError / float(len(points))</code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right; background-color: rgb(238, 238, 238);"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li><li style="box-sizing: border-box; padding: 0px 5px;">3</li><li style="box-sizing: border-box; padding: 0px 5px;">4</li></ul>

现在问题被转化为,寻找参数 b m ,使得误差函数 Error(b,m) 有最小值。在这里, xi yi 都被视为已知值。从下图看,最小二乘法所做的是通过数学推导直接计算得到最低点;而梯度下降法所做的是从图中的任意一点开始,逐步找到图的最低点。

这里写图片描述

2. 多元线性回归模型

从机器学习的角度来说,以上的数据只有一个feature,所以用一元线性回归模型即可。这里我们将一元线性模型的结论一般化,即推广到多元线性回归模型。这部分内部参考了机器学习中的数学(1)-回归(regression)、梯度下降(gradient descent)。假设有 x1 x2 ... xn n 个feature, θ x 的系数,则

 hθ(x)=θ0+θ1x1+...+θnxn=θTxx0=1
 J(θ)=12i=1m(hθ(x(i))y(i))2mm

更一般地,我们可以得到广义线性回归。 ϕ(x) 可以换成不同的函数,从而得到的拟合函数就不一定是一条直线了。

广线 hθ(x)=θTx=θ0+i=1nθiϕi(xi)

2.1 误差函数的进一步思考

这里有一个有意思的东西,就是误差函数为什么要写成这样的形式。首先是误差函数最前面的系数 12 ,这个参数其实对结果并没有什么影响,这里之所以取 12 ,是为了抵消求偏导过程中得到的 2 。可以实验,把 Error(b,m) 最前面的 1N 修改或者删除并不会改变最终的拟合结果。那么为什么要使用平方误差呢?考虑以下公式:

y(i)=θTx(i)+ε(i)

假定误差 ε(i)(1im) 是独立同分布的,由中心极限定理可得, ε(i) 服从均值为 0 ,方差为 σ2 的正态分布(均值若不为0,可以归约到 θ0 )。进一步的推导来自从@邹博_机器学习的机器学习课件。

图片名称 
图片名称

所以求 maxL(θ) 的过程,就变成了求 minJ(θ) 的过程,从理论上解释了误差函数 J(θ) 的由来。

3 最小二乘法求误差函数最优解

最小二乘法(normal equation)相信大家都很熟悉,这里简单进行解释并提供python实现。首先,我们进一步把 J(θ) 写成矩阵的形式。 X m n 列的矩阵(代表 m 个样本,每个样本有 n 个feature), θ Y m 1 列的矩阵。所以 

 J(θ)=12i=1m(hθ(x(i))y(i))2=12(XθY)T(XθY)

这里写图片描述

所以 θ 的最优解为: θ=(XTX)1XTY

当然这里可能遇到一些问题,比如 X 必须可逆,比如求逆运算时间开销较大。具体解决方案待补充。

3.1 python实现最小二乘法

这里的代码仅仅针对背景里的这个问题。部分参考了回归方法及其python实现

<code class="hljs python has-numbering" style="display: block; padding: 0px; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-radius: 0px; word-wrap: normal; background: transparent;"><span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;"># 通过最小二乘法直接得到最优系数,返回计算出来的系数b, m</span>
<span class="hljs-function" style="box-sizing: border-box;"><span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">def</span> <span class="hljs-title" style="box-sizing: border-box;">least_square_regress</span><span class="hljs-params" style="color: rgb(102, 0, 102); box-sizing: border-box;">(points)</span>:</span>
    x_mat = np.mat(np.array([np.ones([len(points)]), points[:, <span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">0</span>]]).T)  <span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;"># 转为100行2列的矩阵,2列其实只有一个feature,其中x0恒为1</span>
    y_mat = points[:, <span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">1</span>].reshape(len(points), <span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">1</span>)  <span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;"># 转为100行1列的矩阵</span>
    xT_x = x_mat.T * x_mat
    <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">if</span> np.linalg.det(xT_x) == <span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">0.0</span>:
        print(<span class="hljs-string" style="color: rgb(0, 136, 0); box-sizing: border-box;">'this matrix is singular,cannot inverse'</span>)  <span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;"># 奇异矩阵,不存在逆矩阵</span>
        <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">return</span>
    coefficient_mat = xT_x.I * (x_mat.T * y_mat)
    <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">return</span> coefficient_mat[<span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">0</span>, <span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">0</span>], coefficient_mat[<span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">1</span>, <span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">0</span>] <span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;"># 即系数b和m</span></code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right; background-color: rgb(238, 238, 238);"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li><li style="box-sizing: border-box; padding: 0px 5px;">3</li><li style="box-sizing: border-box; padding: 0px 5px;">4</li><li style="box-sizing: border-box; padding: 0px 5px;">5</li><li style="box-sizing: border-box; padding: 0px 5px;">6</li><li style="box-sizing: border-box; padding: 0px 5px;">7</li><li style="box-sizing: border-box; padding: 0px 5px;">8</li><li style="box-sizing: border-box; padding: 0px 5px;">9</li><li style="box-sizing: border-box; padding: 0px 5px;">10</li></ul>

程序执行结果如下: 
b = 7.99102098227, m = 1.32243102276, error = 110.257383466, 相关系数 = 0.773728499888

拟合结果如下图:

图片名称

4. 梯度下降法求误差函数最优解

有了最小二乘法以后,我们已经可以对数据点进行拟合。但由于最小二乘法需要计算 X 的逆矩阵,计算量很大,因此特征个数多时计算会很慢,只适用于特征个数小于100000时使用;当特征数量大于100000时适合使用梯度下降法。最小二乘法与梯度下降法的区别见最小二乘法和梯度下降法有哪些区别?

4.1. 梯度

首先,我们简单回顾一下微积分中梯度的概念。这里参考了方向导数与梯度,具体的证明请务必看一下这份材料,很短很简单的。

讨论函数 z=f(x,y) 在某一点 P 沿某一方向的变化率问题。设函数 z=f(x,y) 在点 P(x,y) 的某一邻域 U(P) 内有定义,自点 P 引射线 l 到点 P(x+Δx,y+Δy) PU(P) ,如下图所示。

这里写图片描述

定义函数 z=f(x,y) 在点 P 沿方向 l 的方向导数为:

fl=limρ0f(x+Δx,y+Δy)f(x,y)ρρ=(Δx)2+(Δy)2

方向导数可以理解为,函数 z=f(x,y) 沿某个方向变化的速率。可以类比一下函数 y=kx+b 的斜率 k=dydx 。斜率越大,函数 y 增长得越快。那么现在问题来了,函数 z=f(x,y) 在点 P 沿哪个方向增加的速度最快?而这个方向就是梯度的方向

gradf(x,y)=fxi+fyj

从几何角度来理解,函数 z=f(x,y) 表示一个曲面,曲面被平面 z=c 截得的曲线在 xoy 平面上投影如下图,这个投影也就是我们所谓的等高线。

这里写图片描述

函数 z=f(x,y) 在点 P(x,y) 处的梯度方向与点 P 的等高线 f(x,y)=c 在这点的法向量的方向相同,且从数值较低的等高线指向数值较高的等高线。

4.2 梯度方向计算

理解了梯度的概念之后,我们重新回到1. 背景中提到的例子。1. 背景提到,梯度下降法所做的是从图中的任意一点开始,逐步找到图的最低点。那么现在问题来了,从任意一点开始, b m 可以往任意方向”走”,如何可以保证我们走的方向一定是使误差函数 Error(b,m) 减小且减小最快的方向呢?回忆4.1. 梯度中提到的结论,梯度的方向是函数上升最快的方向,那么函数下降最快的方向,也就是梯度的反方向。有了这个思路,我们首先计算梯度方向,

Error(b,m)m=i=1Nxi((b+mxi)yi)
Error(b,m)b=i=1N((b+mxi)yi)x01

有了这两个结果,我们就可以开始使用梯度下降法来寻找误差函数 Error(b,m) 的最低点。我们从任意的点 (b,m) 开始,逐步地沿梯度的负方向改变 b m 的值。每一次改变, Error(b,m) 都会得到更小的值,反复进行该操作,逐步逼近 Error(b,m) 的最低点。

回到更一般的情况,对于每一个向量 θ 的每一维分量 θi ,我们都可以求出梯度的方向,也就是错误函数 J(θ) 下降最快的方向: 

θjJ(θ)=θj12i=1m(hθ(x(i))y(i))2=i=1m(hθ(x(i))y(i))x(i)j

4.3 批量梯度下降法

从上面的公式中,我们进一步得到特征的参数 θj 的迭代式。因为这个迭代式需要把m个样本全部带入计算,所以我们称之为批量梯度下降

θj=θjαJ(θ)θj=θjαi=1m(hθ(x(i))y(i))x(i)j

针对此例,梯度下降法一次迭代过程的python代码如下:

<code class="hljs python has-numbering" style="display: block; padding: 0px; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-radius: 0px; word-wrap: normal; background: transparent;"><span class="hljs-function" style="box-sizing: border-box;"><span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">def</span> <span class="hljs-title" style="box-sizing: border-box;">step_gradient</span><span class="hljs-params" style="color: rgb(102, 0, 102); box-sizing: border-box;">(b_current, m_current, points, learningRate)</span>:</span>
    b_gradient = <span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">0</span>
    m_gradient = <span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">0</span>
    N = float(len(points))
    <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">for</span> i <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">in</span> range(<span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">0</span>, len(points)):
        x = points[i, <span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">0</span>]
        y = points[i, <span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">1</span>]
        m_gradient += (<span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">2</span> / N) * x * ((b_current + m_current * x) - y)
        b_gradient += (<span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">2</span> / N) * ((b_current + m_current * x) - y)
    new_b = b_current - (learningRate * b_gradient)  <span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;"># 沿梯度负方向</span>
    new_m = m_current - (learningRate * m_gradient)  <span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;"># 沿梯度负方向</span>
    <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">return</span> [new_b, new_m]</code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right; background-color: rgb(238, 238, 238);"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li><li style="box-sizing: border-box; padding: 0px 5px;">3</li><li style="box-sizing: border-box; padding: 0px 5px;">4</li><li style="box-sizing: border-box; padding: 0px 5px;">5</li><li style="box-sizing: border-box; padding: 0px 5px;">6</li><li style="box-sizing: border-box; padding: 0px 5px;">7</li><li style="box-sizing: border-box; padding: 0px 5px;">8</li><li style="box-sizing: border-box; padding: 0px 5px;">9</li><li style="box-sizing: border-box; padding: 0px 5px;">10</li><li style="box-sizing: border-box; padding: 0px 5px;">11</li><li style="box-sizing: border-box; padding: 0px 5px;">12</li></ul>

其中learningRate是学习速率,它决定了逼近最低点的速率。可以想到的是,如果learningRate太大,则可能导致我们不断地最低点附近来回震荡; 而learningRate太小,则会导致逼近的速度太慢。An Introduction to Gradient Descent and Linear Regression提供了完整的实现代码GradientDescentExample

这里多插入一句,如何在python中生成GIF动图。配置的过程参考了使用Matplotlib和Imagemagick实现算法可视化与GIF导出。需要安装ImageMagick,使用到的python库是Wand: a ctypes-based simple ImageMagick binding for Python。然后修改C:\Python27\Lib\site-packages\matplotlib__init__.py文件,在

<code class="hljs actionscript has-numbering" style="display: block; padding: 0px; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-radius: 0px; word-wrap: normal; background: transparent;"># <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">this</span> <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">is</span> the instance used by the matplotlib classes
rcParams = rc_params()</code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right; background-color: rgb(238, 238, 238);"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li></ul>

后面加上:

<code class="hljs vala has-numbering" style="display: block; padding: 0px; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-radius: 0px; word-wrap: normal; background: transparent;"><span class="hljs-preprocessor" style="color: rgb(68, 68, 68); box-sizing: border-box;"># fix a bug by ZZR</span>
rcParams[<span class="hljs-string" style="color: rgb(0, 136, 0); box-sizing: border-box;">'animation.convert_path'</span>] = <span class="hljs-string" style="color: rgb(0, 136, 0); box-sizing: border-box;">'C:\Program Files\ImageMagick-6.9.2-Q16\convert.exe'</span></code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right; background-color: rgb(238, 238, 238);"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li></ul>

即可在python中调用ImageMagick。如何画动图参见Matplotlib动画指南,不再赘述。

learningRate=0.0001,迭代100轮的结果如下图:

After {100} iterations b = 0.0350749705923, m = 1.47880271753, error = 112.647056643, 相关系数 = 0.773728499888 
After {1000} iterations b = 0.0889365199374, m = 1.47774408519, error = 112.614810116, 相关系数 = 0.773728499888 
After {1w} iterations b = 0.607898599705, m = 1.46754404363, error = 112.315334271, 相关系数 = 0.773728499888 
After {10w} iterations b = 4.24798444022, m = 1.39599926553, error = 110.786319297, 相关系数 = 0.773728499888

这里写图片描述
这里写图片描述

4.4 随机梯度下降法

批量梯度下降法每次迭代都要用到训练集的所有数据,计算量很大,针对这种不足,引入了随机梯度下降法。随机梯度下降法每次迭代只使用单个样本,迭代公式如下:

θj=θjα(hθ(x(i))y(i))x(i)j

可以看出,随机梯度下降法是减小单个样本的错误函数,每次迭代不一定都是向着全局最优方向,但大方向是朝着全局最优的。

这里还有一些重要的细节没有提及,比如如何确实learningRate,如果判断何时递归可以结束等等。

参考文献

  1. An Introduction to Gradient Descent and Linear Regression
  2. 方向导数与梯度
  3. 最小二乘法和梯度下降法有哪些区别?
  4. GradientDescentExample
  5. 机器学习中的数学(1)-回归(regression)、梯度下降(gradient descent)
  6. @邹博_机器学习
  7. 回归方法及其python实现
  8. 使用Matplotlib和Imagemagick实现算法可视化与GIF导出
  9. Wand: a ctypes-based simple ImageMagick binding for Python
  10. Matplotlib动画指南

  • 0
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值