zoukankan      html  css  js  c++  java
  • Python实现线性回归2,梯度下降算法

    接上篇

    4.梯度下降算法

    《斯坦福大学公开课 :机器学习课程》吴恩达讲解第二课时,是直接从梯度下降开始讲解,最后采用向量和矩阵的方式推导了解析解,国内很多培训视频是先讲解析解后讲梯度下降,个人认为梯度下降算法更为重要,它是很多算法(逻辑回归、神经网络)都可以共用的,线性回归仅仅是凑巧有全局最小值的解法,有解析解,其他大部分算法都需要遍历数据寻找全局(其实本质上是局部最优)最优解的过程。
    解析解直接向量运算存在的问题:(1)矩阵必须是满秩的,如果不是,python也能模糊处理。(2)运算性能,矩阵扩大化后,量级增大对性能影响明显。
    梯度下降不仅限于线性回归,非线性和神经网络同样适用。

    在线性回归中,θ通过α学习速率在每个J( θ) 减小的维度上进行一个 不定式 θ参数的递减,不断更新 θ{i} 的值直到 J(θ) 收敛,达到最小值。类比于下山,每次尝试找比这个位置低的位置,一步一步到达山底(即我们求的最小值,详细请看参考资料8)

    4.1批量梯度下降算法

    老师常常教导我们梯度下降算法这么经典的算法,一定要自己动手推导公式,自己用程序来实现以下,我一直嗤之以鼻,线性回归这么简单的算法还需要自己敲代码实现一下吗?最近工作闲暇时,我自己用Python实现了下该算法,不敲不知道,一敲吓一大跳,太多坑在里面了,整整坑在里面2天时间,检查多遍代码最后发现是数据问题,学习率α需要取非常非常小的值才能收敛,否则程序一直处于震荡状态无法找到近似最小值。说了那么多废话,下面直接贴代码,代码里备注说明很完善,很明了,很清晰了,大家应该能看懂吧o( ̄︶ ̄)o
    在这里插入图片描述
    在这里插入图片描述

    def linearRegBgd2(x,y,alpha=0.005,lamba=0.005,loop_max=1000):
        #alpha = 0.0000001      #学习率步长,learning rate
        #lamba=0.005     #惩罚系数 
        '''
        参考链接:https://blog.csdn.net/mango_badnot/article/details/52328740 
        已将原作者计算过程进行了大量修改,就相当于是我原创了吧,将原计算的三重for循环重写为一重for循环,将运算修改为了矩阵运算
        原作者构造的X特征是按行存储的,与常见数据不太一致,X中第一行为特征1,第二行为特征2
        此函数将使用正常数据,X特征按列存储,每行是一条记录
    
        批量梯度下降算法公式:
        theta=theta + alpha * sum( (y-y_hat) * x)
        '''
        np.set_printoptions(precision=4) # 设置numpy输出4位小数
        n=len(x[0]) # 取第一行数据,查看数据列数
        theta=np.zeros(n)      # 初始化theta数组,默认全为0
            
        for times in range(loop_max):
            y_hat=np.dot(x,theta.T).reshape((-1,1))
            loss= (y_hat-y)  #此处是y_hat-y,对应的theta求解处增加了一个负号
            loss_n=np.array([]) # 为方便直接用矩阵实现sum( (y-y_hat) * x),产生多列loss乘以loss后求和
            if n==1: # 判断x矩阵有几列,列为1的时候单独处理
                loss_n=loss
            else:
                for i in range(1,n):# 判断x矩阵有几列,列为大于2的时候,产生n列loss
                    #print(i)
                    if i==1:
                        loss_n = np.column_stack((loss, loss))
                    elif i>1:
                        loss_n = np.column_stack((loss_n, loss))
            
            theta_old=theta # 记录迭代前的theta
            #tmp=alpha*(loss_n*x).sum(axis=0)
    
            theta=theta - (alpha*(x*loss_n)).sum(axis=0) #+lamba*theta   
            #使用矩阵求解theta,注意此处的负号,loss*x后按特征列求和作为theta,注意上方本来想实现惩罚项的,但没有想明白怎么实现
            
            if (theta-theta_old).all()<0.001: # 判断前后两次theta变换情况,当变化率很小时跳出循环
                break
            #print('x:',x,'y:',y,'loss:',loss)
            #print('y_hat:',y_hat.T)
            #print('times:',times,'theta:',theta)
            #print('')
        return theta
    

    4.2随机梯度下降算法

    随机梯度下降就是每一个样本更新一次权值,超过样本范围就循环从第一个样本再循环,同等数据量情况下(数据量不是特别巨大时)训练速度与批量梯度下降要慢一些,但是适合在线学习,来一条数据迭代训练一次。
    在这里插入图片描述

    def linearRegSgd(x,y,alpha=0.005,lamba=0.005,loop_max=10000):
        #alpha = 0.0000001      #学习率步长,learning rate
        #lamba=0.005     #惩罚系数 
        '''
        随机梯度下降算法公式:
        theta=theta + alpha * (y-y_hat) * x
        alpha=0.01
        lamba=0.005
        '''
        np.set_printoptions(precision=4) # 设置numpy输出4位小数
        n=len(x[0]) # 取第一行数据,查看数据列数
        theta=np.zeros(n)      # 初始化theta数组,默认全为0
            
        for times in range(loop_max):
            for i  in range(0,len(x)): # 取其中一条数据进行梯度下降
                # i=0
                y_hat=np.dot(x[i],theta.T).reshape((-1,1))
                loss= (y_hat-y[i])  #此处是y_hat-y,对应的theta求解处增加了一个负号
                theta_old=theta # 记录迭代前的theta
                theta=theta - alpha*x[i]*loss[0] #+lamba*theta
                #求解theta,注意此处的负号,注意上方本来想实现惩罚项的,但没有想明白怎么实现
                if (theta-theta_old).all()<0.001: # 判断前后两次theta变换情况,当变化率很小时跳出循环
                    break
    #            print('x:',x,'y:',y,'loss:',loss)
    #            print('y_hat:',y_hat.T)
    #            print('times:',times,'theta:',theta)
    #            print('')
        return theta
    

    4.3 mini-batch梯度下降算法

    如果不是每拿到一个样本即更改梯度,而是用若干个样本的平均梯度作为更新方向,这就是Mini-batch梯度下降算法。

    def linearRegMgd(x,y,alpha=0.005,lamba=0.005,loop_max=1000,batch_size=9):
        #alpha = 0.0000001      #学习率步长,learning rate
        #lamba=0.005     #惩罚系数 
        '''
        mini-batch梯度下降算法公式:
        每次使用batch_size个数据进行计算
        for i=1 to m: 
            theta=theta + alpha * (y-y_hat) * x
        '''
        np.set_printoptions(precision=4) # 设置numpy输出4位小数
        n=len(x[0]) # 取第一行数据,查看数据列数
        theta=np.zeros(n)      # 初始化theta数组,默认全为0
        
        for times in range(loop_max):
            for i in range(0,int(len(x)/batch_size)+1):
                #print('i:',i,x[i*batch_size:(i+1)*batch_size])
                x_mini=x[i*batch_size:(i+1)*batch_size]
                y_mini=y[i*batch_size:(i+1)*batch_size]
                
                y_hat=np.dot(x_mini,theta.T).reshape((-1,1))
                loss= (y_hat-y_mini)  #此处是y_hat-y,对应的theta求解处增加了一个负号
                loss_n=np.array([]) # 为方便直接用矩阵实现sum( (y-y_hat) * x),产生多列loss乘以loss后求和
                if n==1: # 判断x矩阵有几列,列为1的时候单独处理
                    loss_n=loss
                else:
                    for i in range(1,n):# 判断x矩阵有几列,列为大于2的时候,产生n列loss
                        #print(i)
                        if i==1:
                            loss_n = np.column_stack((loss, loss))
                        elif i>1:
                            loss_n = np.column_stack((loss_n, loss))
                
                theta_old=theta # 记录迭代前的theta
                #tmp=alpha*(loss_n*x).sum(axis=0)
        
                theta=theta - (alpha*(x_mini*loss_n)).sum(axis=0) #+lamba*theta   
                #使用矩阵求解theta,注意此处的负号,loss*x后按特征列求和作为theta,注意上方本来想实现惩罚项的,但没有想明白怎么实现
                
                if (theta-theta_old).all()<0.001: # 判断前后两次theta变换情况,当变化率很小时跳出循环
                    break
        
            #print('x:',x,'y:',y,'loss:',loss)
            #print('y_hat:',y_hat.T)
            #print('times:',times,'theta:',theta)
            #print('')
        return theta
    

    以上梯度下降算法完整代码如下

    # -*- coding: utf-8 -*-
    """
     @Time    : 2018/10/22 17:23
     @Author  : hanzi5
     @Email   : **@163.com
     @File    : linear_regression_bgd.py
     @Software: PyCharm
    """
    import numpy as np
    import pandas as pd
    import matplotlib.pyplot as plt
    from sklearn.linear_model import LinearRegression
    
    def load_dataset(n): 
        # 原作者产生数据的函数
        noise = np.random.rand(n)
        X = [[x, 1.] for x in range(n)]
        y = [(0.5 * X[i][0]  + 1. + noise[i]) for i in range(n)]
        return np.array(X).T, np.array(y).T # 注意X,W,y的维数
    
    def linearRegLsq(x,y):
        # 最小二乘法直接求解theta
        xtx = np.dot(x.T, x)
        if np.linalg.det(xtx) == 0.0: # 判断xtx行列式是否等于0,奇异矩阵不能求逆
            print('Can not resolve the problem')
            return
        theta_lsq = np.dot(np.dot(np.linalg.inv(np.dot(x.T, x)), x.T), y)
        return theta_lsq
    
    def linearRegBgd1(x,y,alpha=0.005,lamba=0.005,loop_max=1000):
        #alpha = 0.0000001      #学习率步长,learning rate
        #lamba=0.005     #惩罚系数 
        '''
        已将原作者计算过程进行了大量修改,已将原作者计算过程进行了大量修改,就相当于是我原创了吧,将原计算的三重for循环重写为一重for循环
        参考链接:https://blog.csdn.net/mango_badnot/article/details/52328740 
        原作者构造的X特征是按行存储的,与常见数据不太一致,X中第一行为特征1,第二行为特征2
        批量梯度下降算法公式:
        theta=theta + alpha * sum( (y-y_hat) * x)
        2018-10-22 17:18 测试通过,此函数不做多的备注了,详细备注放在了linearRegBgd2里
        '''
        
        np.set_printoptions(precision=4) # 设置numpy输出4位小数
        n=len(x) # 取第一行数据,查看数据列数
        theta=np.zeros(n)      # 初始化theta数组,默认全为0
        for times in range(loop_max):
            y_hat=np.dot(x.T,theta)
            loss= y-y_hat
            for i in range(1,n):
                if i==1:
                    loss_n = np.row_stack((loss, loss))
                elif i>1:
                    loss_n = np.row_stack((loss_n, loss))
            
            theta_old=theta
            theta=theta+(alpha*x.T*loss_n.T).sum(axis=0) #+lamba*theta
            if (theta-theta_old).all()<0.001:
                break
            #print('x:',x,'y:',y,'loss:',loss)
            #print('times:',times,'theta:',theta)
            #print('')
        return theta
    
    def linearRegBgd2(x,y,alpha=0.005,lamba=0.005,loop_max=1000):
        #alpha = 0.0000001      #学习率步长,learning rate
        #lamba=0.005     #惩罚系数 
        '''
        参考链接:https://blog.csdn.net/mango_badnot/article/details/52328740 
        已将原作者计算过程进行了大量修改,就相当于是我原创了吧,将原计算的三重for循环重写为一重for循环,将运算修改为了矩阵运算
        原作者构造的X特征是按行存储的,与常见数据不太一致,X中第一行为特征1,第二行为特征2
        此函数将使用正常数据,X特征按列存储,每行是一条记录
    
        批量梯度下降算法公式:
        theta=theta + alpha * sum( (y-y_hat) * x)
        '''
        np.set_printoptions(precision=4) # 设置numpy输出4位小数
        n=len(x[0]) # 取第一行数据,查看数据列数
        theta=np.zeros(n)      # 初始化theta数组,默认全为0
            
        for times in range(loop_max):
            y_hat=np.dot(x,theta.T).reshape((-1,1))
            loss= (y_hat-y)  #此处是y_hat-y,对应的theta求解处增加了一个负号
            loss_n=np.array([]) # 为方便直接用矩阵实现sum( (y-y_hat) * x),产生多列loss乘以loss后求和
            if n==1: # 判断x矩阵有几列,列为1的时候单独处理
                loss_n=loss
            else:
                for i in range(1,n):# 判断x矩阵有几列,列为大于2的时候,产生n列loss
                    #print(i)
                    if i==1:
                        loss_n = np.column_stack((loss, loss))
                    elif i>1:
                        loss_n = np.column_stack((loss_n, loss))
            
            theta_old=theta # 记录迭代前的theta
            #tmp=alpha*(loss_n*x).sum(axis=0)
    
            theta=theta - (alpha*(x*loss_n)).sum(axis=0) #+lamba*theta   
            #使用矩阵求解theta,注意此处的负号,loss*x后按特征列求和作为theta,注意上方本来想实现惩罚项的,但没有想明白怎么实现
            
            if (theta-theta_old).all()<0.001: # 判断前后两次theta变换情况,当变化率很小时跳出循环
                break
            #print('x:',x,'y:',y,'loss:',loss)
            #print('y_hat:',y_hat.T)
            #print('times:',times,'theta:',theta)
            #print('')
        return theta
    
    def linearRegSgd(x,y,alpha=0.005,lamba=0.005,loop_max=10000):
        #alpha = 0.0000001      #学习率步长,learning rate
        #lamba=0.005     #惩罚系数 
        '''
        随机梯度下降算法公式:
        for i=1 to m: 
            theta=theta + alpha * (y-y_hat) * x
        
        alpha=0.01
        lamba=0.005
        '''
        np.set_printoptions(precision=4) # 设置numpy输出4位小数
        n=len(x[0]) # 取第一行数据,查看数据列数
        theta=np.zeros(n)      # 初始化theta数组,默认全为0
            
        for times in range(loop_max):
            for i  in range(0,len(x)): # 取其中一条数据进行梯度下降
                # i=0
                y_hat=np.dot(x[i],theta.T).reshape((-1,1))
                loss= (y_hat-y[i])  #此处是y_hat-y,对应的theta求解处增加了一个负号
                theta_old=theta # 记录迭代前的theta
                theta=theta - alpha*x[i]*loss[0] #+lamba*theta
                #求解theta,注意此处的负号,注意上方本来想实现惩罚项的,但没有想明白怎么实现
                if (theta-theta_old).all()<0.001: # 判断前后两次theta变换情况,当变化率很小时跳出循环
                    break
    #            print('x:',x,'y:',y,'loss:',loss)
    #            print('y_hat:',y_hat.T)
    #            print('times:',times,'theta:',theta)
    #            print('')
        return theta
    
    def linearRegMgd(x,y,alpha=0.005,lamba=0.005,loop_max=1000,batch_size=9):
        #alpha = 0.0000001      #学习率步长,learning rate
        #lamba=0.005     #惩罚系数 
        '''
        mini-batch梯度下降算法公式:
        每次使用batch_size个数据进行计算
        for i=1 to m: 
            theta=theta + alpha * (y-y_hat) * x
        '''
        np.set_printoptions(precision=4) # 设置numpy输出4位小数
        n=len(x[0]) # 取第一行数据,查看数据列数
        theta=np.zeros(n)      # 初始化theta数组,默认全为0
        
        for times in range(loop_max):
            for i in range(0,int(len(x)/batch_size)+1):
                #print('i:',i,x[i*batch_size:(i+1)*batch_size])
                x_mini=x[i*batch_size:(i+1)*batch_size]
                y_mini=y[i*batch_size:(i+1)*batch_size]
                
                y_hat=np.dot(x_mini,theta.T).reshape((-1,1))
                loss= (y_hat-y_mini)  #此处是y_hat-y,对应的theta求解处增加了一个负号
                loss_n=np.array([]) # 为方便直接用矩阵实现sum( (y-y_hat) * x),产生多列loss乘以loss后求和
                if n==1: # 判断x矩阵有几列,列为1的时候单独处理
                    loss_n=loss
                else:
                    for i in range(1,n):# 判断x矩阵有几列,列为大于2的时候,产生n列loss
                        #print(i)
                        if i==1:
                            loss_n = np.column_stack((loss, loss))
                        elif i>1:
                            loss_n = np.column_stack((loss_n, loss))
                
                theta_old=theta # 记录迭代前的theta
                #tmp=alpha*(loss_n*x).sum(axis=0)
        
                theta=theta - (alpha*(x_mini*loss_n)).sum(axis=0) #+lamba*theta   
                #使用矩阵求解theta,注意此处的负号,loss*x后按特征列求和作为theta,注意上方本来想实现惩罚项的,但没有想明白怎么实现
                
                if (theta-theta_old).all()<0.001: # 判断前后两次theta变换情况,当变化率很小时跳出循环
                    break
        
            #print('x:',x,'y:',y,'loss:',loss)
            #print('y_hat:',y_hat.T)
            #print('times:',times,'theta:',theta)
            #print('')
        return theta
    
    if __name__ == "__main__":
        path = 'D:/python_data/8.Advertising.csv'
        data = pd.read_csv(path)  # TV、Radio、Newspaper、Sales
        #data_matrix = data.as_matrix(columns=['TV', 'Radio', 'Newspaper', 'Sales'])  # 转换数据类型
        data_array = data.values[:,1:]  # 转换数据类型,去除第一列序号列
        x = data_array[:, :-1] #去除y对应列的数据,其他列作为x
        y = data_array[:, -1].reshape((-1,1))
    
        # 0、绘制图像,查看数据分布情况
        plt.plot(data['TV'], y, 'ro', label='TV')
        plt.plot(data['Radio'], y, 'g^', label='Radio')
        plt.plot(data['Newspaper'], y, 'mv', label='Newspaer')
        plt.legend(loc='lower right')
        plt.grid()
        plt.show()
        
        # 1、使用sklearn LinearRegression包求解θ
        linreg = LinearRegression()
        model = linreg.fit(x, y)
        print('')
        print('1、sklearn LinearRegression包求解θ:','coef:', linreg.coef_[0],',intercept:', linreg.intercept_)
    
        # 2、最小二乘法,直接求解析解θ
        theta_lsq = linearRegLsq(x,y)
        print('')
        print('2、最小二乘法,theta解析解:',theta_lsq.reshape(1,3)[0])
        
        # 3、批量梯度下降求解theta
        # 注意下面两个函数alpha都是非常小的值,取过大的值时,不收敛
        #x1, y1 = load_dataset(10)
        #theta1=linearRegBgd1(x1, y1)
        
        theta1=linearRegBgd1(x.T, y.T,alpha = 0.0000001)
        print('')
        print('3.1、批量梯度下降,linearRegBgd1函数,theta:',theta1)
        
        theta2=linearRegBgd2(x, y,alpha = 0.0000001)
        print('')
        print('3.2、批量梯度下降,linearRegBgd2函数,theta:',theta2)
        
        theta3=linearRegSgd(x, y,alpha = 0.000001,loop_max=10000)
        print('')
        print('3.3、随机梯度下降,linearRegSgd函数,theta:',theta3)
        
        theta4=linearRegMgd(x, y,alpha = 0.000001,loop_max=10000,batch_size=11)
        print('')
        print('3.4、mini-batch梯度下降,linearRegMgd函数,theta:',theta4)
        
    

    程序运行计算结果:
    在这里插入图片描述

    数据见以下链接,复制到txt文档中,另存为CSV格式即可。
    数据

    参考资料:
    1、python实现线性回归
    2、机器学习:线性回归与Python代码实现
    3、python实现简单线性回归
    4、Machine-Learning-With-Python
    5、《机器学习》西瓜书,周志华
    6、机器学习视频,邹博
    7、 斯坦福大学公开课 :机器学习课程
    8、马同学高等数学 如何直观形象的理解方向导数与梯度以及它们之间的关系?
    9、【机器学习】线性回归的梯度下降法

  • 相关阅读:
    【Python3网络爬虫开发实战】3.1.2-处理异常
    02018_StringBuffer练习
    富文本编辑器可以如何直接复制word的图文内容到编辑器中?
    TinyMCE可以如何直接复制word的图文内容到编辑器中?
    wangEditor可以如何直接复制word的图文内容到编辑器中?
    xhEditor可以如何直接复制word的图文内容到编辑器中?
    FCKEditor可以如何直接复制word的图文内容到编辑器中?
    KindEditor可以如何直接复制word的图文内容到编辑器中?
    CKEditor可以如何直接复制word的图文内容到编辑器中?
    百度编辑器可以如何直接复制word的图文内容到编辑器中?
  • 原文地址:https://www.cnblogs.com/hanzi5/p/10105622.html
Copyright © 2011-2022 走看看