Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 754|回复: 0

机器学习中梯度下降法原理及用其解决线性回归问题的C语言实现

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-4-27 13:41:12 | 显示全部楼层 |阅读模式

    本文讲梯度下降(Gradient Descent)前先看看利用梯度下降法进行监督学习(例如分类、回归等)的一般步骤:

    1, 定义损失函数(Loss Function)

    2, 信息流forward propagation,直到输出端

    3, 误差信号back propagation。采用“链式法则”,求损失函数关于参数Θ的梯度

    4, 利用最优化方法(比如梯度下降法),进行参数更新

    5, 重复步骤2、3、4,直到收敛为止

    所谓损失函数,就是一个描述实际输出值和期望输出值之间落差的函数。有多种损失函数的定义方法,常见的有均方误差(error of mean square)、最大似然误差(maximum likelihood estimate)、最大后验概率(maximum posterior probability)、交叉熵损失函数(cross entropy loss)。本文就以均方误差作为损失函数讲讲梯度下降的算法原理以及用其解决线性回归问题。在监督学习下,对于一个样本,它的特征记为x(如果是多个特征,x表示特征向量),期望输出记为t(t为target的缩写),实际输出记为o(o为output的缩写)。两者之间的误差e可用下式表达(为了节省时间,各种算式就用手写的了):

    前面的系数1/2主要是为了在求导时消掉差值的平方项2。如果在训练集中有n个样本,可用E来表示所有样本的误差总和,并用其大小来度量模型的误差程度,如下式所示:

     对于第d个实例的输出可记为下式:

    对于特定的训练数据集而言, 只有Θ是变量,所以E就可以表示成Θ的函数,如下式:

    所以,对于神经网络学习的任务,就是求到一系列合适的Θ值,以拟合给定的训练数据,使实际输出尽可能接近期望输出,使得E取得最小值。

     

    再来看梯度下降。上式中损失函数E对权值向量Θ的梯度如下式所示:

    它确定了E最快上升的方向。在梯度前面加上负号“-”,就表示E最快下降的方向。所以梯度下降的训练法则如下式所示:

    , 其中

    这里的负号“-”表示和梯度相反的方向。η表示学习率。下面给出各个权值梯度计算的数学推导:

    所以最终的梯度下降的训练法则如下式:

        

     这个式子就是用于程序中计算参数Θ的。

    下面看怎么用梯度下降法解决线性回归问题。线性回归就是能够用一个直线较为精确地描述数据之间的关系。这样当出现新的数据的时候,就能够预测出一个简单的值。线性回归函数可写成 。线性回归问题常用最小二乘法解决,这里用梯度下降法解决主要是通过实例加深对梯度下降法的理解。先假设Y = 2X + 3=2*X + 3*1,取X的四个值分别为1,4,5,8,相应的Y为5,11,13,19。这样就可以描述为有四个样本分别为(1,1),(4,1),(5,1),(8,1),对应的期望值是5,11,13,19.5(这个值做了微调,从19变成了19.5,是为了让四个样本不在一根直线上)。通过梯度下降法求Θ值(最终Θ逼近2和3)。C语言实现的代码如下:

    #include <stdio.h>
    #include <stdlib.h>

    int
    main(int argc, char *argv[]) { double matrix[4][2]={{1,1},{4,1},{5,1},{8,1}}; //样本 double result[4]={5,11,13,19.5}; //期望值 double err_sum[4] = {0,0,0,0}; //各个样本的误差 double theta[2] = {1,6}; //Θ,初始值随机 double err_square_total = 0.0; //方差和 double learning_rate = 0.01; //学习率 int ite_num; //迭代次数 for(ite_num = 0; ite_num <= 10000; ite_num++) { int i,j,k; err_square_total = 0.0; for(i = 0; i < 4; i++) { double h = 0; for(j = 0; j < 2; j++) h += theta[j]*matrix[j]; err_sum = result - h; err_square_total += 0.5*err_sum*err_sum; } if(err_square_total < 0.05) //0.05表示精度 break; for(j = 0; j < 2; j++) { double sum = 0; for(k = 0; k < 4; k++) //所有样本都参与计算 sum += err_sum[k]*matrix[k][j]; theta[j] = theta[j] + learning_rate*sum; //根据上面的公式计算新的Θ } } printf(" @@@ Finish, ite_number:%d, err_square_total:%lf, theta[0]:%lf, theta[1]:%lf\n", ite_num, err_square_total, theta[0], theta[1]); return 0; }

    程序运行后的结果为:@@@ Finish, ite_number:308, err_square_total:0.049916, theta[0]:2.037090, theta[1]:3.002130。发现迭代了308次,最终的线性方程为Y=2.037090X + 3.002130,是逼近2和3的。当再有一个新的X时就可以预测出Y了。学习率是一个经验值,一般是0.01--0.001,当我把它改为0.04再运行时就不再收敛了。

     

    上面的梯度下降叫批量梯度下降法(Batch Gradient Descent, BGD), 它是指在每一次迭代时使用所有样本来进行梯度的更新当样本数目很大时,每迭代一步都需要对所有样本计算,训练过程会很慢。于是人们想出了随机梯度下降法(Stochastic Gradient Descent, SGD),每次只随机取一个样本计算梯度,训练速度变快了,但是迭代次数变多了(表示不是一直向最快方向下降,但总体上还是向最低点逼近)。还是上面的例子,只不过每次只从四个样本中随机取一个计算梯度。C语言实现的代码如下:

    #include <stdio.h>
    #include <stdlib.h>


    int
    main(int argc, char *argv[]) { double matrix[4][2]={{1,1},{4,1},{5,1},{8,1}}; //样本 double result[4]={5,11,13,19.5}; //期望值 double err_sum[4] = {0,0,0,0}; //各个样本的误差 double theta[2] = {1,6}; //Θ,初始值随机 double err_square_total = 0.0; //方差和 double learning_rate = 0.01; //学习率 int ite_num; //迭代次数 for(ite_num = 0; ite_num <= 10000; ite_num++) { int i,j,seed; err_square_total = 0.0; for(i = 0; i < 4; i++) { double h = 0;
    for(j = 0; j < 2; j++) h += theta[j]*matrix[j];
    err_sum
    = result - h; err_square_total += 0.5*err_sum*err_sum; } if(err_square_total < 0.05) break; seed = rand()%4; for(j = 0; j < 2; j++) theta[j] = theta[j] + learning_rate*err_sum[seed]*matrix[seed][j]; //随机选一个样本参与计算 } printf(" @@@ Finish, ite_number:%d, err_square_total:%lf, theta[0]:%lf, theta[1]:%lf\n", ite_num, err_square_total, theta[0], theta[1]); return 0; }

    程序运行后的结果为:@@@ Finish, ite_number:1228, err_square_total:0.049573, theta[0]:2.037240, theta[1]:3.000183。发现迭代了1228次(迭代次数变多了),最终的线性方程为Y=2.037240X + 3.000183,也是逼近2和3的。

     

    后来人们又想出了在BGD和SGD之间的一个折中方法,即mini-batch SGD方法,即每次随机的取一组样本来计算梯度。mini-batch SGD是实际使用中用的最多的。还是上面的例子,只不过每次只从四个样本中随机取两个作为一组个计算梯度。C语言实现的代码如下:

    #include <stdio.h>
    #include <stdlib.h>

    int
    main(int argc, char *argv[]) { double matrix[4][2]={{1,1},{4,1},{5,1},{8,1}}; double result[4]={5,11,13,19.5}; double err_sum[4] = {0,0,0,0}; double theta[2] = {1,6}; double err_square_total = 0.0; double learning_rate = 0.01; int ite_num; for(ite_num = 0; ite_num <= 10000; ite_num++) { int i,j,k,seed; err_square_total = 0.0;
    for(i = 0;i<4;i++) { double h = 0;
    for(j = 0; j < 2; j++) h += theta[j]*matrix[j]; err_sum = result - h; err_square_total += 0.5*err_sum*err_sum; } if(err_square_total < 0.05) break; seed = rand()%4; k = (seed +1)%4; for(j = 0; j < 2; j++) { double sum = 0; sum += err_sum[seed]*matrix[seed][j]; //随机取两个作为一组计算梯度 sum += err_sum[k]*matrix[k][j];
    theta[j]
    = theta[j] + learning_rate*sum; } } printf(" @@@ Finish, ite_number:%d, err_square_total:%lf, theta[0]:%lf, theta[1]:%lf\n", ite_num, err_square_total, theta[0], theta[1]); return 0; }

    程序运行后的结果为: @@@ Finish, ite_number:615, err_square_total:0.047383, theta[0]:2.039000, theta[1]:2.987382。发现迭代了615次,最终的线性方程为Y=2.039000X + 2.987382,也是逼近2和3的。迭代次数介于BGD和SGD中间。在用mini-batch SGD时batch size的选择很关键。

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-1-5 09:58 , Processed in 0.061106 second(s), 29 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表