• 一 准备

    • 1 数据集
    • 2 基本工具
      • 21 pandasread in data
      • 22 numpyprocess data
      • 23 matplotlibvisualize data
  • 二 基本概念与定义
  • 三 感知机学习算法的原始形式以误分数为损失函数
  • 四 基于最优化方法的变体梯度下降随机梯度下降
  • 五 python实战
    • 1 类接口设计
    • 2 感知机的基本形式以误分数为损失函数
    • 3 平方误差之和损失函数下的感知机的梯度下降解
      • 31 考察learning rate的影响
      • 32 feature 标准化
    • 4 感知机的随机梯度下降解
  • 六 总结

感知机算法的基本形式及一些基于最优化方法的感知机算法的变体具有机器学习的典型处理框架,且理论较为简单,实现并不复杂。同时作为单层神经网络的感知机,也是支持向量机和神经网络的基础。

本文的主要内容包括:

  • 为什么说通过感知机算法的推导和实践便迈进了机器学习的大门?
  • 感知机算法的学习策略是怎样的?
  • 不同感知机算法之间的真正区别是什么?
  • 关于感知机算法都有哪些损失函数和权重更新的形式?
  • 为什么需要对数据进行标准化或者叫特征缩放?
  • 如何对数据进行标准化?
  • 如何利用python实现感知机算法及其不同的变体?
  • 如何利用python(matplotlib)实现对数据的可视化工作?

一、 准备

本文算法的任务是通过应用感知机算法及其不同最优化方法下(GD:梯度下降,SGD:随机梯度下降)的版本实现对一个线性可分的二维(也可应用到多维,无非是np.dot(x, w[1:]), x和w维度的不同,对算法的实现没有任何影响,只是出于可视化的需要)二类别样本的分类,这中间涉及一些简单的数据处理(比如standardized,标准化,或者叫特征缩放,feature scaling,又比如数据shuffle),以及更为重要的通过matplotlib实现数据可视化,来一窥机器学习算法的全貌。

1.1 数据集

本文待分类的数据集(iris.data)是一种CSV(逗号分隔符,Comma-Separated Value)文件。
点击这里可查看对数据集的介绍,这里对其中的属性介绍如下:

  1. sepal length in cm
  2. sepal width in cm
  3. petal length in cm
  4. petal width in cm
  5. class:
    – Iris Setosa
    – Iris Versicolour
    – Iris Virginica

为了说明问题的方便,本文只涉及二分类问题,考虑前100个样本,即Iris-setosaIris-versicolor类的样本各占一半,为了低维数据可视化的需要,考虑第一和第三个属性,即sepal lengthpetal length

import matplotlib.pyplot as plt
plt.scatter(X[y==1, 0], X[y==1, 1], color='red', marker='o', label='setosa')
plt.scatter(X[y==-1, 0], X[y==-1, 1], color='blue', marker='x', label='versicolor')

这100个样本的散列图如下:

注:iris,鸢尾花。鸢尾花张啥样(这才是真正的可视化呀)?

1.2 基本工具

1.2.1 pandas——read in data

import pandas as pd
  • pandas文件读取函数
df = pd.read_csv('https://archive.ics.uci.edu/ml/''machine-learning-databases/iris/iris.data', header=None)   # 返回一种DataFrame结构文件
print(df.tail())                    # 验证是否读取正确
  • DataFrame,pandas中的一种数据结构

通过pd.read_csv()读取到待处理的数据集并保存到DataFrame后,便可通过DataFrame强大的切片和索引能力,生成机器学习算法所需要的数据格式。

# 只考虑第一和第三个属性值,为了可视化的的需要
# 此时X的类型是numpy.ndarray
X = df.iloc[[0:100, [0, 2]].values()
y = df.iloc[[0:100], 4].values
# 将string类型的类别标签转化为二分类标签
y = np.where(y == 'Iris-seosta', -1, 1)

1.2.2 numpy——process data

1.2.3 matplotlib——visualize data

二、 基本概念与定义

感知机是二分类的线性模型,以实例的特征向量(不同于线性代数的特征向量,其实线代里的特征向量更准确的叫法应该是本征向量)作为输入,以±1\pm 1类别标签作为输出。

输入空间(input space 或者叫 feature space)X⊆Rn\mathcal X \subseteq \mathbb R^n,输出空间(output space)Y=+1,−1\mathcal Y = {+1, -1}。
我们定义激励函数(activation function,ϕ(z)\phi(z)),它以输入x与权重w的内积作为输入,z=wTx=w0x0+w1x1+w2x2+…+wdxdz=w^Tx=w_0x_0+w_1x_1+w_2x_2+\ldots+w_dx_d,也就是所谓的net_input(或者叫score),以最终的预测类别作为输出.

感知机算法中,ϕ(⋅)\phi(\cdot)是一个简单的单位阶跃函数(有时也称作heaviside step function):

ϕ(z)={+1,−1,z≥0otherwise

\begin{eqnarray} \phi(z)= \begin{cases} +1, &z\geq 0\cr -1, &otherwise \end{cases} \end{eqnarray}

注:关于bias的两种解释,解释1:w1x1+w2x2+…+wdxd+w0∗1w_1x_1+w_2x_2+\ldots+w_dx_d+w_0*1,如下图(三层神经网络)示,这里的偏置用来表征与输入无关的一些系统自身的因素,更具数学意味的说法是将数据的输入空间提升了一维。

解释之二,在激励函数(activation function)中,对输进行了加权求和之后的net_input(或者叫score),如果大于指定的阈值 θ\theta,则输出为1,否则为-1

z=w1x1+w2x2+…+wdxdϕ(z)={+1,−1,z≥θotherwise

\begin{eqnarray} & z = w_1x_1+w_2x_2+\ldots+w_dx_d\\ & \phi(z)= \begin{cases} +1, &z\geq \theta \cr -1, &otherwise \end{cases} \end{eqnarray}
转化为如下的形式:

z=w0x0+w1x1+w2x2+…+wdxd=wTxϕ(z)={+1,−1,z≥0otherwise

\begin{eqnarray} & z = w_0x_0+ w_1x_1+w_2x_2+\ldots+w_dx_d = w^Tx\\ & \phi(z)= \begin{cases} +1, &z\geq 0\cr -1, &otherwise \end{cases} \end{eqnarray}
其中 w0=−θw_0=-\theta, x0=1x_0=1。

殊途同归,两中不同的解释得到相同的形式。

三、 感知机学习算法的原始形式(以误分数为损失函数)

以误分数作为损失函数的形式如下:

J=∑i=1N1y(i)≠ϕ(wTx(i))

J = \sum_{i=1}^{N} 1_{y^{(i)} \neq \phi(w^Tx^{(i)})}

其权重更新方式如下:

w:=w+ΔwΔ=η(y(i)−y^(i))x(i)

w:=w+\Delta w\\ \Delta=\eta (y^{(i)}-\hat{y}^{(i)})x^{(i)}

其中,η\eta表示学习率,是一个介于0-1之间的常量,控制更新的步长,在classifier构造时定义。

以二维数据集为例:

Δw0=Δw1=Δw2=η(y(i)−y^(i))η(y(i)−y^(i))x(i)1η(y(i)−y^(i))x(i)2

\begin{eqnarray} \Delta w_0=&\eta(y^{(i)}-\hat {y}^{(i)})\\ \Delta w_1=&\eta(y^{(i)}-\hat {y}^{(i)})x^{(i)}_1\\ \Delta w_2=&\eta(y^{(i)}-\hat {y}^{(i)})x^{(i)}_2\\ \end{eqnarray}
注意,这里的 Δwj\Delta w_j是按照向量化的做法同步得到的,也即:

self.w_[0] += self.eta * (y - self.predict(x))
self.w_[1:] += self.eta * (y-self.predict(x)) * x
# 这里的self.w_[1:]以及x表示的均是向量

在我们进入python实战之前不妨,先对这一权值更新方式的合理性做一个直观的解释,为什么这样的权值更新是可行的。
监督学习的过程中,真实值和估计值之间差值(y(i)−y^(i)y^{(i)}-\hat{y}^{(i)})一共四种情况,
Δw=η(1−1)x(i)=0\Delta w=\eta(1-1)x^{(i)} = 0
Δw=η(−1−−1)x(i)=0\Delta w=\eta(-1--1)x^{(i)} = 0
Δw=η(1−−1)x(i)=η(2)x(i)\Delta w=\eta(1--1)x^{(i)} = \eta (2)x^{(i)}
Δw=η(−1−1)x(i)=η(−2)x(i)\Delta w=\eta(-1-1)x^{(i)} = \eta (-2)x^{(i)}

在误分的情况下,所进行的权值更新,以Δw=η(1−−1)x(i)=η(2)x(i)\Delta w=\eta(1--1)x^{(i)} = \eta (2)x^{(i)}为例(将正的估计成了负的),根据上述的权值更新公式得到的更新的方向(+2),恰代表真实值所在的方向。

四、 基于最优化方法的变体(梯度下降、随机梯度下降)

这一节,我们将介绍单层神经网络的另一种实现方法:自适应线性神经元(ADAptive LInear NEuron,AdaLine)。
AdaLine与Rosenblatt’s 基本感知机的关键不同在于,两个具有不同的损失函数及相关的权值更新公式。后者是基于单位阶跃函数(y(i)−y^(i)y^{(i)}-\hat{y}^{(i)}),而前者是基于线性激励函数(yi−ϕ(wTx)y^{i}-\phi(w^Tx))。在AdaLine中,线性激励函数ϕ(wTx)\phi(w^Tx)是简单的证同函数,即ϕ(wTx)=wTx\phi(w^Tx)=w^Tx。

监督学习算法的一大关键在于目标函数(objective function)的定义与设计,目标函数是学习或者训练的过程(learning process)中进行优化的对象。目标函数常常是我们需要最小化的损失函数。在AdaLine算法中,我们可以通过将损失函数JJ定义为平方误差和(sum-squared-error,SSE)的方式来进行对权值的更新,

J==12∑i=1N(y(i)−ϕ(z(i)))212∥y−Xw∥2

\begin{eqnarray} J = &\frac12\sum_{i=1}^{N}(y^{(i)}-\phi (z^{(i)}))^2\\ =&\frac 12 \|y-Xw\|^2 \end{eqnarray}

12\frac 12是为了求导的方便。与单位阶跃函数相比,这种线性激励函数的最大优势在于,损失函数变成了可导的,而且是凸函数,因此我们可以利用梯度下降法找到权值更新的方向来最小化损失函数,即:

w:=Δw=w+Δw−ηΔJ(w)

\begin{eqnarray} &w:=&w+\Delta w\\ &\Delta w=&-\eta \Delta J(w) \end{eqnarray}

因为是找使损失函数最小的方向,故权值更新的方向是负梯度方向(−ηΔJ(w)-\eta\Delta J(w))
由损失函数的矩阵形式,我们很容易求出它的梯度:

∂J∂w=−XT(y−Xw)

\frac {\partial J}{\partial w} = -X^T(y-Xw)

所以新的权值更新方式:

w=w+ηXT(y−Xw)

w=w+\eta X^T(y-Xw)

五、 python实战

5.1 类接口设计

根据机器学习清晰的两相处理流程(学习或者叫训练,以及预测),再加之面向对象的思想,可对分类器设计如下的接口。

  • fit(X, y)

接受样本集X,及样本类别标签y,返回分类器实例

  • net_input(X)

接受样本集X,返回样本集与权重的内积X⋅wX\cdot w(X列代表属性feature,行代表一个观察值observation,w是列向量),但这里要考虑bias的作用,也就是如果Xw都不是增广的话,X⋅w[1:]+w[0]X\cdot w[1:]+w[0]。

  • activation(X)

如:

ϕ(z)={+1,−1,z≥θotherwise

\begin{eqnarray} \phi(z)= \begin{cases} +1, &z\geq \theta \cr -1, &otherwise \end{cases} \end{eqnarray}

  • predict(X)

接受样本集X,返回预测的类别标签±1\pm 1。如果判别函数是y=sign(wtx)y=sign(w^tx),则其python实现形式如下:

return np.where(self.net_predict(X) >= 0., 1, -1)

5.2 感知机的基本形式——以误分数为损失函数

class Perceptron(object):def __init__(self, eta=0.01, n_iter=10):self.eta = etaself.n_iter = n_iterdef fit(self, X, y):self.w_ = np.zeros(1 + X.shape[1])self.J_ = []        # 记录每次迭代的误分数for _ in range(self.n_iter):J = 0for xi, yi in zip(X, y):delta = yi - self.activation(xi)                self.w_[0] += self.eta * deltaself.w_[1:] += self.eta * delta * xiJ += int(delta != 0.)self.J_.append(J)return selfdef net_input(self, X):return X.dot(self.w_[1:]) + self.w_[0]def activation(self, X):# python中的三目运算符return np.where(self.net_input(X) >= 0.0, 1, -1)    def predict(self, X):return np.where(self.net_input(X) >= 0.0, 1, -1)

客户端程序,并显示随着迭代的进行,误分数的变化情况:

ppn = Perceptron(eta=0.1, n_iter=10)
ppn.fit(X, y)
plt.plot(range(1, 1 + len(ppn.J_), ppn.J_, marker='o')
plt.xlabel('epochs')
plt.ylabel('# of iterations')
plt.show()

收敛效果图

可视化

from matplotlib.colors import ListedColormapdef plot_decision_regions(X, y, classifier, res=0.02):markers = ('s', 'o', 'x', '^', 'v')colors = ('red', 'blue', 'lightgreen', 'gray', 'cyan')colormap = ListedColormap(colors[:len(np.unique(y))])x1_min, x1_max = X[:, 0].min() - 1, X[:, 0].max() + 1x2_min, x2_max = X[:, 1].min() - 1, X[:, 1].max() + 1xx1, xx2 = np.meshgrid(np.arange(x1_min, x1_max, res), np.arange(x2_min, x2_max, res))Z = classifier.predict(np.array([xx1.ravel(), xx2.ravel()]).T)# xx1.ravel() 将xx1从numpy.narray类型的多维素组转换为一位数组Z = Z.reshpae(xx1.shape())np.contourf(xx1, xx2, Z, alpha=.4, cmap=colormap)plt.xlim(x1_min, x1_max)plt.ylim(x2_max, x2_max)for idx, cl in enumerate(np.unique(y)):plt.scatter(x=X[y==cl, 0], y=X[y==cl, 1], marker=markers[idx],alpha=.8, cmap=colormap(idx), label=np.where(cl==1, 'versicolor', 'setosa'))
ppn = Perceptron(eta=0.1, n_iter=10).fit(X, y)
plot(X, y, ppn, res=0.02)
plt.xlabel('sepal length [cm]')
plt.ylabel('petal length [cm]')
plt.legend(loc='upper right')
plt.show()

5.3 平方误差之和损失函数下的感知机的梯度下降解

class AdaLine(object):def __init__(self, eta=0.01, n_iter=10):self.eta = etaself.n_iter = n_iterdef fit(self, X, y):self.w_ = np.zeros(1+X.shape[1])        self.J_ = []for _ in range(self.n_iter):errors = y - self.activation(X)self.w_[0] += self.eta * errors.sum()self.w_[1:] += self.eta * X.T.dot(errors)J = errors.dot(errors)/2.self.J_.append(J)def net_input(self, X):return X.dot(self.w_[1:]) + self.w_[0]def activation(self, X):return net_input(X)def predict(self, X):return np.where(self.net_input(X) >= 0., 1, -1)

5.3.1 考察learning rate的影响

我们来考虑两个不同学习率下(η=0.01,η=0.0001\eta = 0.01, \eta = 0.0001)的损失函数收敛情况:

fig, ax = plt.subplots(nrows=1, ncols=2, figsize=(8, 4))    # 一行两列的子图分布
ada1 = AdaLine(eta=0.01, n_iter=10).fit(X, y)
ax[0].plot(range(1, 1+len(ada1.J_)), np.log10(ada1.J_)), marker='o')
ax[0].set_xlabel('epochs')
ax[0].set_ylabel('sse')
ax[0].set_title('learning rate: 0.01')
ada2 = AdaLine(eta=0.0001, n_iter=10)
ax[1].plot(range(1, 1+len(ada2.J_)), ada2.J_, marker='o')
ax[1].set_xlabel('epochs')
ax[1].set_ylabel('sse')
ax[1].set_title('learning rate: 0.0001')
plt.show()

5.3.2 feature 标准化

在以上的两图中我们可以明显的看出,因为学习率选择的过大,损失函数错过了全局最小值,发生了发散,不再收敛(对应于左图),而学习率选择的过小,导致收敛速度很慢。

这里我们需要对输入特征进行缩放(feature scaling),或者叫标准化(standardization):

x′j=xj−μjσj

x_j^{'} = \frac {x_j-\mu_j}{\sigma_j}

特征的标准化工作大量地用在包括梯度下降在内的很多机器学习算法中。经标准化后的算法的执行结果如下:

5.4 感知机的随机梯度下降解

5.3节的算法因为进行梯度更新时考虑的是一次性的将整个样本集都计算在内,

w=w+ηXT(y−XTw)

w=w+\eta X^T(y-X^Tw)
故有时也称作块梯度下降(batch gradient descent)。试想我们要处理的是一个更大规模的数据集,处理大规模数据集是机器学习算法所要面对的普遍状况。块梯度下降算法将会十分的耗时,因为每次迭代都要对整个数据集做计算。
块梯度算法的一种流行的替代方案是随机梯度下降(stochastic gradient descent),有时也叫iterative 或者 on-line gradient descent。

随机梯度下降与块梯度算法的不同在于,后者更新梯度时基于的是全部样本的加权和:

Δw=ηXT(y−Xw)

\Delta w=\eta X^T(y-Xw)

前者则是针对每一个样本,进行增量式地更新权值,即:

Δw=η(y(i)−ϕ(wTx(i)))x(i)

\Delta w=\eta (y^{(i)}-\phi(w^Tx^{(i)}))x^{(i)}

from numpy.random import seedclass AdaLineSGD(object):def __init__(self, eta=0.01, n_iter=10, shuffle=True, random_state=None):self.eta = etaself.n_iter = n_iterself.shuffle = shuffleself.w_initilized = Falseif self.shuffle:seed(random_state)def fit(self, X, y)self.w_ = self._initilized_weights(X.shpae[1])      self.J_ = []for _ in range(self.n_iter):if self.shuffle:X, y = self._shuffle(X, y)J = 0for xi, yi in zip(X, y):error = yi - self.activation(xi)self.w_[1:] += self.eta * error * xiself.w_[0] += self.eta * errorJ += error**2self.J_.append(J/2./len(y))return self def net_input(self, X):return X.dot(self.w_[1:]) + self.w_[0]def activation(self, X):return self.net_input(X)def _initilized_weights(self, d):self.w_initilized = Truereturn np.zeros(1 + d)def _shuffle(self, X, y):r = np.random.permulation(X.shape[0])return X[r, :], y[r]def predict(self, X):return self.net_input(X)

应用SGD算法的分类情况及损失函数的收敛情况如下:

六、 总结

algorithm 代价函数 权值更新
感知机基本形式 J=∑Ni=11y(i)≠ϕ(wTx(i))J = \sum_{i=1}^{N} 1_{y^{(i)} \neq \phi(w^Tx^{(i)})} Δw=η(y(i)−y^(i))x(i)\Delta w=\eta (y^{(i)}-\hat{y}^{(i)})x^{(i)}
梯度下降 J=12∥y−Xw∥2J=\frac12\|y-Xw\|^2 Δw=ηXT(y−Xw)\Delta w=\eta X^T(y-Xw)
随机梯度下降 J=12∥y−Xw∥2J=\frac12\|y-Xw\|^2 Δw=η(y(i)−ϕ(wTx(i)))x(i)\Delta w=\eta(y^{(i)}-\phi(w^Tx^{(i)}))x^{(i)}

python 机器学习——从感知机算法到各种最优化方法的应用(python)相关推荐

  1. python机器学习手写算法系列——逻辑回归

    从机器学习到逻辑回归 今天,我们只关注机器学习到线性回归这条线上的概念.别的以后再说.为了让大家听懂,我这次也不查维基百科了,直接按照自己的理解用大白话说,可能不是很严谨. 机器学习就是机器可以自己学 ...

  2. python机器学习手写算法系列——线性回归

    本系列另一篇文章<决策树> https://blog.csdn.net/juwikuang/article/details/89333344 本文源代码: https://github.c ...

  3. python机器学习手写算法系列——kmeans聚类

    从机器学习到kmeans 聚类是一种非监督学习,他和监督学习里的分类有相似之处,两者都是把样本分布到不同的组里去.区别在于,分类分析是有标签的,聚类是没有标签的.或者说,分类是有y的,聚类是没有y的, ...

  4. Python 机器学习/深度学习/算法专栏 - 导读目录

    目录 一.简介 二.机器学习 三.深度学习 四.数据结构与算法 五.日常工具 一.简介 Python 机器学习.深度学习.算法主要是博主从研究生到工作期间接触的一些机器学习.深度学习以及一些算法的实现 ...

  5. 大数据基石python学习_资源 | 177G Python/机器学习/深度学习/算法/TensorFlow等视频,涵盖入门/中级/项目各阶段!...

    原标题:资源 | 177G Python/机器学习/深度学习/算法/TensorFlow等视频,涵盖入门/中级/项目各阶段! 这是一份比较全面的视频教程,基本上包括了市面上所有关于机器学习,统计学习, ...

  6. python 预测算法_通过机器学习的线性回归算法预测股票走势(用Python实现)

    本文转自博客园,作者为hsm_computer 原文链接:https://www.cnblogs.com/JavaArchitect/p/11717998.html在笔者的新书里,将通过股票案例讲述P ...

  7. 机器学习之感知机算法

    简介 感知机是一种较为简单的二分类模型,但由简至繁,感知机却是神经网络和支持向量机的基础.感知机旨在学习能够将输入数据划分为+1/-1的线性分离超平面,所以说整体而言感知机是一种线性模型.因为是线性模 ...

  8. 用Python解决数据结构与算法问题(一):Python基础

    python学习之路 - 从入门到精通到大师 文章目录 [python学习之路 - 从入门到精通到大师](https://blog.csdn.net/TeFuirnever/article/detai ...

  9. 通过机器学习的线性回归算法预测股票走势(用Python实现)

    在本人的新书里,将通过股票案例讲述Python知识点,让大家在学习Python的同时还能掌握相关的股票知识,所谓一举两得.这里给出以线性回归算法预测股票的案例,以此讲述通过Python的sklearn ...

最新文章

  1. 使用 HTML5、CSS3 和 MathML 在 EPUB 3 中制作版式丰富的出版物
  2. 现在python已经更新到哪个版本了-Python 3.8 已发布 你会升级么?
  3. am335x PDK3.0 设置为单网口配置记录
  4. webpack常用loader和plugin及打包速度优化
  5. 内联元素(display:inline-block)注意事项
  6. Spring Boot参数校验
  7. 列车时刻管理c语言程序设计,列车时刻表信息管理系统实践报告C语言源代码
  8. mysql 表锁的概念_MySQL 锁的一些简单概念
  9. 价格要大涨,新5G iPhone可能不会带来换机热?
  10. xampp配置虚拟主机
  11. 移除collection中元素的注意事项(应用collection.remove移除元素造成的错误)
  12. java get 和post_[Java教程]get和post方法的区别
  13. 模拟电路---利用反相器构成方波电路在实际电路中的使用
  14. [渝粤教育] 西南交通大学 工程流体力学 参考 资料
  15. MDM主数据清洗和编码集成说明
  16. 四川大学计算机网络实验,四川大学计算机网络实验报告
  17. 单击选定单元格后输入新内容_excel表格,要双击选定后才能输入,单击不能输入为什么...
  18. JavaSE基础笔记——JOptionPane编写员工管理系统;GUI使用;写一个超级数组
  19. Eclipse代码/目录虚线对齐设置
  20. 网站被黑总被跳转到其他网站的处理解决过程

热门文章

  1. 详解:离线项目二 每个区域下最受欢迎的产品TOPN
  2. php 动态修改函数,php – 动态调用函数
  3. debian 查看php版本,在debian 9中为php7.0-fpm启用状态查询
  4. html5试卷答案,常见的HTML5前端面试题及答案
  5. Java依赖多个版本,java – Gradle中的多个依赖项版本
  6. 数据挖掘:银行评分卡制作——数据分箱、WOE、IV的意义
  7. Bootstrap-Table事件和方法
  8. ISR4K-IOS XE EPC
  9. 基于Laravel+VueJS实战开发WebAPP
  10. 临时和永久关闭Selinux