《李航:统计学习方法》笔记之感知机

2024-06-09 17:38

本文主要是介绍《李航:统计学习方法》笔记之感知机,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

感知机学习旨在求出将训练数据集进行线性划分的分类超平面,为此,导入了基于误分类的损失函数,然后利用梯度下降法对损失函数进行极小化,从而求出感知机模型。感知机模型是神经网络和支持向量机的基础。也是现代流行的深度学习网络模型的基础。下面分别从感知机学习的模型、策略和算法三个方面来介绍。

1. 感知机模型

      感知机模型如下:

f(x)= sign(w*x+b)

      其中,x为输入向量,sign为符号函数,括号里面大于等于0,则其值为1,括号里面小于0,则其值为-1。w为权值向量,b为偏置。求感知机模型即求模型参数w和b。感知机预测,即通过学习得到的感知机模型,对于新的输入实例给出其对应的输出类别1或者-1。

2. 感知机策略

      假设训练数据集是线性可分的,感知机学习的目标就是求得一个能够将训练数据集中正负实例完全分开的分类超平面,为了找到分类超平面,即确定感知机模型中的参数w和b,需要定义一个损失函数并通过将损失函数最小化来求w和b。

       这里选择的损失函数是误分类点到分类超平面S的总距离。输入空间中任一点x 0到超平面S的距离为:

其中,||w||为w的L2范数。

        其次,对于误分类点来说,当-y (wx i + b)>0时,y i=-1,当-y i(wx i + b)<0时,y i=+1。所以对误分类点(x i, y i)满足:

-y(wxi +b) > 0

所以误分类点(x i, y i)到分类超平面S的距离是:

3. 感知机算法

       感知机学习问题转化为求解损失函数式(1)的最优化问题,最优化的方法是随机梯度下降法。感知机学习算法是误分类驱动的,具体采用随机梯度下降法。首先,任意选取一个超平面w0,b0,然后用梯度下降法不断极小化目标函数式(1)。极小化的过程不是一次使M中所有误分类点的梯度下降,而是一次随机选取一个误分类点使其梯度下降。

       损失函数L(w,b)的梯度是对w和b求偏导,即:

其中,(0<<=1)是学习率,即学习的步长。综上,感知机学习算法如下:

        这种算法的基本思想是:当一个实例点被误分类,即位于分类超平面错误的一侧时,则调整w和b,使分类超平面向该误分类点的一侧移动,以减少该误分类点与超平面的距离,直到超平面越过该误分类点使其被正确分类为止。

       需要注意的是,这种感知机学习算法得到的模型参数不是唯一的,它会由于采用不同的参数初始值或选取不同的误分类点,而导致解不同。为了得到唯一的分类超平面,需要对分类超平面增加约束条件,线性支持向量机就是这个想法。另外,当训练数据集线性不可分时,感知机学习算法不收敛,迭代结果会发生震荡。而对于线性可分的数据集,算法一定是收敛的,即经过有限次迭代,一定可以得到一个将数据集完全正确划分的分类超平面及感知机模型。

       以上是感知机学习算法的原始形式,下面介绍感知机学习算法的对偶形式,对偶形式的基本想法是,将w和b表示为实例x i和标记y i的线性组合形式,通过求解其系数而求得w和b。对误分类点(x i, y i)通过


所以,感知机学习算法的对偶形式如下:

感知机的原始形式和对偶形式在解决问题的计算上是一致的,但是他们的思想不同,原始形式的基本思想是对于误分类点,调整w和b,使分类超平面向该误分类点的一侧移动,以减少该误分类点与超平面的距离,直到超平面越过该误分类点使其被正确分类为止。 而对偶形式的基本思想是将w和b表示成x和y的线性组合形式,从而求出w和b。 

1)原始形式代码如下:

[cpp]  view plain copy
  1. #include <iostream>  
  2. using namespace std;  
  3.   
  4. int x[3][2] = {  
  5.     {3, 3},  
  6.     {4, 3},  
  7.     {1, 1}  
  8. };  
  9.   
  10. int y[3] = {1, 1, -1};  
  11.   
  12. int w[2] = {0};  
  13. int b = 0;  
  14.   
  15. int L(int y, int* x)  
  16. {  
  17.     int temp = (w[0] * x[0] + w[1] * x[1] + b) * y;  
  18.     if (temp <= 0)  
  19.         return 1;//存在错误点  
  20.     else  
  21.         return 0;  
  22. }  
  23.   
  24. int main(void)  
  25. {  
  26.     int j = 1;  
  27.     while (true)  
  28.     {  
  29.         cout << j++ << " ";  
  30.           
  31.         int i;  
  32.         int num = 0;  
  33.         for (i = 0; i < 3; i++)  
  34.         {  
  35.             if (L(y[i], x[i]) == 1)  
  36.             {  
  37.                 cout << "error point:";  
  38.                 cout << "x" << i <<" w:";  
  39.                 int j;  
  40.                 for (j = 0; j < 2; j++)  
  41.                 {  
  42.                     w[j] += y[i] * x[i][j];  
  43.                     cout << w[j] << " ";  
  44.                 }  
  45.                 b += y[i];  
  46.                 cout << "b:" << b <<endl;  
  47.                 num++;  
  48.                 break;  
  49.             }  
  50.         }  
  51.         if (num == 0)  
  52.             break;  
  53.     }  
  54.     return 0;  
  55. }  

实验结果:

1 error point:x0 w:3 3 b:1

2 error point:x2 w:2 2 b:0

3 error point:x2 w:1 1 b:-1

4 error point:x2 w:0 0 b:-2

5 error point:x0 w:3 3 b:-1

6 error point:x2 w:2 2 b:-2

7 error point:x2 w:1 1 b:-3

 

这跟p30的结果是一样的,不过要注意的是,在极小化的过程中,为了达到书中的结果,选择的误分类点都是第一次遇到的误分类点,而实际上在选择误分类点时应该采用随机的方法来选取,而且每次梯度下降的时候并不是对所有误分类点进行梯度下降,而是只对随机选择的一个误分类点进行梯度下降。结果与误分类点的选择有关。

2)对偶形式,代码如下:

[cpp]  view plain copy
  1. #include <iostream>  
  2. using namespace std;  
  3.   
  4. int x[3][2] = {  
  5.     {3, 3},  
  6.     {4, 3},  
  7.     {1, 1}  
  8. };  
  9.   
  10. int y[3] = {1, 1, -1};  
  11.   
  12. int b = 0;  
  13. int a[3] = {0};  
  14. int G[3][3] = {  
  15.     {18, 21, 6},  
  16.     {21, 25, 7},  
  17.     {6, 7, 2}  
  18. };//Gram matrix  
  19. int L(int j)  
  20. {  
  21.     int temp = 0;  
  22.     for (int i=0 ;i < 3; i++)  
  23.     {  
  24.         temp += a[i] * G[i][j] * y[i];  
  25.     }  
  26.     temp += b;  
  27.     temp *= y[j];  
  28.     if (temp <= 0)  
  29.         return 1;//存在错误点  
  30.     else  
  31.         return 0;  
  32. }  
  33.   
  34. int main(void)  
  35. {  
  36.     int j = 1;  
  37.     while (true)  
  38.     {  
  39.         cout << j++ << " ";  
  40.           
  41.         int i;  
  42.         int num = 0;  
  43.         for (i = 0; i < 3; i++)  
  44.         {  
  45.             if (L(i) == 1)  
  46.             {  
  47.                 cout << "error point:";  
  48.                 cout << "x" << i <<" a:";  
  49.                 int j;  
  50.                 a[i] += 1;  
  51.                 for (j = 0; j < 3; j++)  
  52.                 {  
  53.                     cout << a[j] << " ";  
  54.                 }  
  55.                 b += y[i];  
  56.                 cout << "b:" << b <<endl;  
  57.                 num++;  
  58.                 break;  
  59.             }  
  60.         }  
  61.         if (num == 0)  
  62.             break;  
  63.     }  
  64.     return 0;  
  65. }  

实验结果如下:

1 error point:x0 a:1 0 0 b:1

2 error point:x2 a:1 0 1 b:0

3 error point:x2 a:1 0 2 b:-1

4 error point:x2 a:1 0 3 b:-2

5 error point:x0 a:2 0 3 b:-1

6 error point:x2 a:2 0 4 b:-2

7 error point:x2 a:2 0 5 b:-3



参考文献:
http://blog.csdn.net/qll125596718/article/details/8394186
http://wenku.baidu.com/link?url=80H_Cz1qBitKGe2IJKwsHqgxS72wMROtDYO9N9NwyV5qbvOJrA9-0sUsWjbCsGO5eEGswZVuUuLD9xlq4A-FRhg3Sb6XrfNws7yvma5ec1y

这篇关于《李航:统计学习方法》笔记之感知机的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1045853

相关文章

Python判断for循环最后一次的6种方法

《Python判断for循环最后一次的6种方法》在Python中,通常我们不会直接判断for循环是否正在执行最后一次迭代,因为Python的for循环是基于可迭代对象的,它不知道也不关心迭代的内部状态... 目录1.使用enuhttp://www.chinasem.cnmerate()和len()来判断for

Java循环创建对象内存溢出的解决方法

《Java循环创建对象内存溢出的解决方法》在Java中,如果在循环中不当地创建大量对象而不及时释放内存,很容易导致内存溢出(OutOfMemoryError),所以本文给大家介绍了Java循环创建对象... 目录问题1. 解决方案2. 示例代码2.1 原始版本(可能导致内存溢出)2.2 修改后的版本问题在

四种Flutter子页面向父组件传递数据的方法介绍

《四种Flutter子页面向父组件传递数据的方法介绍》在Flutter中,如果父组件需要调用子组件的方法,可以通过常用的四种方式实现,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录方法 1:使用 GlobalKey 和 State 调用子组件方法方法 2:通过回调函数(Callb

一文详解Python中数据清洗与处理的常用方法

《一文详解Python中数据清洗与处理的常用方法》在数据处理与分析过程中,缺失值、重复值、异常值等问题是常见的挑战,本文总结了多种数据清洗与处理方法,文中的示例代码简洁易懂,有需要的小伙伴可以参考下... 目录缺失值处理重复值处理异常值处理数据类型转换文本清洗数据分组统计数据分箱数据标准化在数据处理与分析过

Java中Object类的常用方法小结

《Java中Object类的常用方法小结》JavaObject类是所有类的父类,位于java.lang包中,本文为大家整理了一些Object类的常用方法,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. public boolean equals(Object obj)2. public int ha

golang1.23版本之前 Timer Reset方法无法正确使用

《golang1.23版本之前TimerReset方法无法正确使用》在Go1.23之前,使用`time.Reset`函数时需要先调用`Stop`并明确从timer的channel中抽取出东西,以避... 目录golang1.23 之前 Reset ​到底有什么问题golang1.23 之前到底应该如何正确的

Vue项目中Element UI组件未注册的问题原因及解决方法

《Vue项目中ElementUI组件未注册的问题原因及解决方法》在Vue项目中使用ElementUI组件库时,开发者可能会遇到一些常见问题,例如组件未正确注册导致的警告或错误,本文将详细探讨这些问题... 目录引言一、问题背景1.1 错误信息分析1.2 问题原因二、解决方法2.1 全局引入 Element

Python调用另一个py文件并传递参数常见的方法及其应用场景

《Python调用另一个py文件并传递参数常见的方法及其应用场景》:本文主要介绍在Python中调用另一个py文件并传递参数的几种常见方法,包括使用import语句、exec函数、subproce... 目录前言1. 使用import语句1.1 基本用法1.2 导入特定函数1.3 处理文件路径2. 使用ex

Oracle查询优化之高效实现仅查询前10条记录的方法与实践

《Oracle查询优化之高效实现仅查询前10条记录的方法与实践》:本文主要介绍Oracle查询优化之高效实现仅查询前10条记录的相关资料,包括使用ROWNUM、ROW_NUMBER()函数、FET... 目录1. 使用 ROWNUM 查询2. 使用 ROW_NUMBER() 函数3. 使用 FETCH FI

Git中恢复已删除分支的几种方法

《Git中恢复已删除分支的几种方法》:本文主要介绍在Git中恢复已删除分支的几种方法,包括查找提交记录、恢复分支、推送恢复的分支等步骤,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录1. 恢复本地删除的分支场景方法2. 恢复远程删除的分支场景方法3. 恢复未推送的本地删除分支场景方法4. 恢复