- A perceptron takes several binary inputs, x1,x2,… and produces a single binary output
- 基本性质:
- output = 0 if ∑jwjxj ≤ threshold else 1
- Using the bias instead of the threshold, the perceptron rule can be rewritten:
- output = 0 if w⋅x+b ≤ 0 else 0,w和x为向量
- Another way perceptrons can be used is to compute the elementary logical functions we usually think of as underlying computation, functions such as AND, OR, and NAND. We can use networks of perceptrons to compute any logical function at all. The reason is that the NAND gate is universal for computation, that is, we can build any computation up out of NAND gates.
- why not use perceptron
a small change in the weights or bias of any single perceptron in the network can sometimes cause the output of that perceptron to completely flip, say from 0 to 1. That flip may then cause the behaviour of the rest of the network to completely change in some very complicated way.
- the sigmoid neuron has inputs, x1,x2... But instead of being just 0 or 1, these inputs can also take on any values between 0 and 1
- 基本性质
- output = σ(w⋅x+b),σ为sigmoid函数
- σ(z) = 1/(1+e^-z)
-
大致思想
- 定义网络的损失函数(有多种不同的损失函数,这里使用平方损失函数)为:C(w,b) = 1/2n ∑x ∥y(x)−a∥^2,x代表任意训练样本;w和b表示网络的所有权值和偏置值;y(x)为x对应的标签(真值),a为网络求得的x的分类结果(预测值)
- 定义Δv = (Δw0,Δw1,Δw2...,Δb0,Δb1,Δb2...)为权值与偏置的变化(偏移量)
- 则可得:ΔC≈∇C⋅Δv,∇C为损失函数C在当前网络状态下的梯度
- 取 Δv = -η∇C (η 称之为学习率,η越大则更新粒度越大)
- 则可得:ΔC ≈ −η∇C⋅∇C = −η∥∇C∥^2 ≤ 0
- 已知平方损失函数恒大于0,不断的利用1~5更新整个网络中的参数直到ΔC趋近于0(无法再进行学习或学习效果不再显著),即可得网络的最优解(局部或全局)
- 上面6个过程中只有∇C是未知的,只要知道了∇C我们就可以更新整个网络,∇C可以使用下面的BP算法求得
- 求∇C有更传统的方式,就是根据梯度与倒数的定义:∂C/∂wj ≈ (C(w+ϵej)−C(w))/ϵ,但这样每次都需要求一次C,运算量太大
-
stochastic gradient descent (随机梯度下降法,这里的随机指的是随机选取每次参加训练的样本,并使用这些样本求得平均的梯度并更新网络)
estimate the gradient ∇C by computing ∇Cx for a small sample of randomly chosen training inputs. By averaging over this small sample it turns out that we can quickly get a good estimate of the true gradient ∇C∇C, and this helps speed up gradient descent, and thus learning.
-
反向传输算法的功能:计算网络中任意w或b关于损失函数的偏导(∂C/∂w和∂C/∂b),所有的∂C/∂w和∂C/∂b的集合就是上面所需的∇C
-
符号说明
- wljk:第
l
层的第j
个神经元和第l-1
层的第k
个神经元之间的连接权重(l和j表示同一层的神经元);- 向量表示法:wl,w是一个矩阵,表示的是第l层神经元与第l-1层神经元之间的所有权重信息,元素(j,k)即为wljk
- blj:第
l
层第j
个神经元的偏置值; - alj:第
l
层第j
个神经元的输出值(即激活函数的输出值);- alj = σ(∑kwljkal-1k+blj);
- 用向量表示为:al=σ(wlal-1+bl)
- 为了计算a,我们需要计算一个中间值zl,即每个神经元提交给激活函数的值:zl = wlal-1+bl
- 那么可简化a的公式为:al=σ(zl)
- wljk:第
-
计算网络损失函数的两个假设
- 可以通过对批量训练时获得的损失函数值求平均求得全局损失函数值(需要这条假设是因为每次训练时我们只使用了部分数据进行训练)
- 可以通过网络的输出求得损失函数值(这一点很明显,因为我们的损失函数的自变量就是网络的输出)
-
Hadamard(Schur) 积(哈达玛积):
[1 2] ⊙ [3 4] = [1*2 2*4] = [3 8]
-
涉及反向传播的四个公式(证明)
- δLj = ∂C/∂zLj = ∂C/∂aLj · σ′(zLj)
δ是C关于最后一层第j个神经元激活函数输入值 z 的偏导,证明方法很简单:
∂C/∂aLj · ∂aLj/∂zLj = ∂C/∂aLj · σ′(zLj)使用向量表示C关于最后一层的偏导:δL=∇aLC ⊙ σ′(zL)
如果使用平方损失函数则可进一步推导得:δL=(aL - y) ⊙ σ′(zL) - δl = (wl+1)Tδl+1 · σ′(zl)
这个公式表明我们可以通过后一层的参数求得 C 对当前层的偏导,这是反向传播算法的核心,可以从网络后面向前求每一层的δ
- ∂C/∂blj=δlj
C关于偏置b的偏导,与公式1思想一致,只不过1是关于神经元输入的
- ∂C/∂wljk=al-1k δlj
C关于每个w的偏导,3,4中已经涉及到我们需要求的参数
- δLj = ∂C/∂zLj = ∂C/∂aLj · σ′(zLj)
-
network类
# import numpy as np class Network(object): # sizes是个数组,分别表示每层网络神经元的个数 # 例如[784, 30, 10],表明输入层有784个元素,隐藏层有30个元素,输出层有10个元素 def __init__(self, sizes): self.num_layers = len(sizes) # 确定网络的层数 self.sizes = sizes # 各层神经元个数 # randomn用于创建指定形状的数组,其中元素满足标准正态分布 # biases中每一列元素的个数都与sizes对应位置元素值相同 self.biases = [np.random.randn(y, 1) for y in sizes[1:]] # 如果sizes=[1,2,3]则sizes[:-1]=[1,2],sizes[1:]=[2,3], # zip用于合并列表并生成矩阵,但要注意其对行列的处理 # zip(sizes[:-1],sizes[1:]) = [[1,2],[2,3]] self.weights = [np.random.randn(y, x) for x, y in zip(sizes[:-1], sizes[1:])] # 这个函数就是测试(分类)函数,给出测试数据a,当前函数将给出a对于各个类别的概率 def feedforward(self, a): # 可以认为a是输入层的数据,即一张图片各个像素点的值数组 for b, w in zip(self.biases, self.weights): a = sigmoid(np.dot(w, a) + b) return a # 训练函数 def SGD(self, training_data, epochs, mini_batch_size, eta, test_data=None): """Train the neural network using mini-batch stochastic gradient descent. The ``training_data`` is a list of tuples ``(x, y)`` representing the training inputs and the desired outputs. The other non-optional parameters are self-explanatory. If ``test_data`` is provided then the network will be evaluated against the test data after each epoch, and partial progress printed out. This is useful for tracking progress, but slows things down substantially.""" # 如果有测试数据则每次训练完成之后使用测试数据进行测试 if test_data: n_test = len(test_data) n = len(training_data) # epochs为训练迭代的次数,每次epoch都将使用所有的数据 for j in xrange(epochs): # 打乱训练数据并mini_batch_size设定每次参见训练的数据量 random.shuffle(training_data) mini_batches = [ training_data[k:k+mini_batch_size] for k in xrange(0, n, mini_batch_size)] # 使用一个mini_batch进行一次网络参数更新 for mini_batch in mini_batches: self.update_mini_batch(mini_batch, eta) if test_data: print "Epoch {0}: {1} / {2}".format( j, self.evaluate(test_data), n_test) else: print "Epoch {0} complete".format(j) def update_mini_batch(self, mini_batch, eta): """Update the network's weights and biases by applying gradient descent using backpropagation to a single mini batch. The ``mini_batch`` is a list of tuples ``(x, y)``, and ``eta`` is the learning rate.""" # 创建了与b和w结构相同但元素值均为0的变量nabla_b和nabla_w nabla_b = [np.zeros(b.shape) for b in self.biases] nabla_w = [np.zeros(w.shape) for w in self.weights] # 使用一个mini_batch中的样本求δ for x, y in mini_batch: # 获得一个样本对网络的微调量 delta_nabla_b, delta_nabla_w = self.backprop(x, y) # 下面的代码将对mini_batch中每个样本的δ求和,获得最终的δ nabla_b = [nb+dnb for nb, dnb in zip(nabla_b, delta_nabla_b)] nabla_w = [nw+dnw for nw, dnw in zip(nabla_w, delta_nabla_w)] # 更新网络参数nw为梯度,eta/len(mini_batch)为步长, # 这里不使用平方可能是因为平方损失函数的特殊性,或其他,我还不是很确定 self.weights = [w-(eta/len(mini_batch))*nw for w, nw in zip(self.weights, nabla_w)] self.biases = [b-(eta/len(mini_batch))*nb for b, nb in zip(self.biases, nabla_b)] # 使用反向传播算法计算网络中每一层w和b的梯度 # 或者可以粗略的认为是当前状态下每个神经元对损失函数的倒数 def backprop(self, x, y):# x表示的是样本数据(一个样本),y是对应的类别(标签) """Return a tuple "(nabla_b, nabla_w)" representing the gradient for the cost function C_x. "nabla_b" and "nabla_w" are layer-by-layer lists of numpy arrays, similar to "self.biases" and "self.weights".""" # 所有层的 ∇b 和 ∇w nabla_b = [np.zeros(b.shape) for b in self.biases] nabla_w = [np.zeros(w.shape) for w in self.weights] # feedforward activation = x #输入层数据 activations = [x] # 保存每一层每个神经元的输出,第一个层为输入x zs = [] # 保存每一层每个神经元的z,即每个神经元激活函数的输入值 # b,w在这里代表了每一层神经元的偏置和权值 # 下面的for循环用于求上面的 activations 和 zs for b, w in zip(self.biases, self.weights): # 求下一层所有神经元的输入 z z = np.dot(w, activation) + b zs.append(z) # 下一层神经元的输出 a activation = sigmoid(z) activations.append(activation) # backward pass # delta(数组) 是损失函数 C 对最后一层神经元 z 的偏导,这是四个方程中的第一个 delta = self.cost_derivative(activations[-1], y) * sigmoid_prime(zs[-1]) # 这是第三个方程 nabla_b[-1] = delta # 这是第四个方程 nabla_w[-1] = np.dot(delta, activations[-2].transpose()) # 因为最后一层的数据已经在上面求得,所以下面从倒数第二层开始更新整个网络中的权值和偏置值 # 下面的for循环中使用的就是第二个方程,依次从后向前求解每层的∇b 和 ∇w for l in xrange(2, self.num_layers): z = zs[-l] sp = sigmoid_prime(z) # 对非后一层使用第二个方程 delta = np.dot(self.weights[-l + 1].transpose(), delta) * sp # 倒数第l层的偏置b对C的偏导,可以认为是倒数第l层b对C的梯度,第三个方程 nabla_b[-l] = delta # 倒数第l层的权值对C的梯度,第四个方程 nabla_w[-l] = np.dot(delta, activations[-l - 1].transpose()) return (nabla_b, nabla_w) def evaluate(self, test_data): """Return the number of test inputs for which the neural network outputs the correct result. Note that the neural network's output is assumed to be the index of whichever neuron in the final layer has the highest activation.""" test_results = [(np.argmax(self.feedforward(x)), y) for (x, y) in test_data] return sum(int(x == y) for (x, y) in test_results) def cost_derivative(self, output_activations, y): """Return the vector of partial derivatives \partial C_x / \partial a for the output activations.""" return (output_activations-y) #### Miscellaneous functions def sigmoid(z): """The sigmoid function.""" return 1.0/(1.0+np.exp(-z)) def sigmoid_prime(z): """Derivative of the sigmoid function.""" return sigmoid(z)*(1-sigmoid(z))
- 因为sigmoid函数的特点,当输出值趋近于1或0时sigmoid函数的斜率会趋近于0。此时整个网络的学习效率会非常低(梯度消失)。这是sigmoid激活函数的缺点
- cross-entropy 是替代sigmoid做激活函数较为不错的选择,具体原因可以参考。
- 现在神经网络中使用的激活函数一般的输出值都在0
1之间,在训练的时候输出层的输出也在01之间。一般训练的时候有几个标签就有几个输出值。softmax将输出值转化为概率,具体解释放在后面再说。 - softmax还可以解决训练时学习速度降低的问题,具体参考这里
- 权值的初始化一般按照网络的规模使用不同的策略,具体可参考这里
- 在深度学习中一般使用验证集来选择网络的结构、超参数、训练epoch等参数【参考】,不使用测试集来选择这些参数是避免选择的参数刚好匹配测试集,对其他的数据(包括训练集与其他测试集)没有一般性。这与网络的过拟合所担心的问题很相似。这时测试集唯一的用处就是检测网络的泛化能力。
- 定义:the model will work well for the existing data, but will fail to generalize to new situations.
- 比较形象的解释
一般而言模型中的变量越多其所能表示的模式也越多。如果模型变量有限那么很可能出现其对某些数据拟合的很好,但对新的数据就没有更好的表现,这是因为有限的变量只能抽取有限的数据特征,模型只能抽取已有数据中特有的那些共有特征而不能适应(识别)新数据中的特征。即模型抽取的特征只是整个样本空间特征的一个子集(即现有数据特有的那一部分),对于新的特征,模型无法再学习(因为无法记忆)也无法有效的检测。
- 过拟合的一些现象:
- 在训练的过程中有时候训练中的网络关于训练集的cost在不断的下降(说明网络对训练集“匹配”越来越好),然而此时网络对测试集的检测精度一直在某个值上下波动,这说明网络队对测试集数据已经没有更好的泛化能力。此时网络的学习已经没有什么效果了,学得再多也只是更好的适配训练集(可能使网络的泛化能力越来越差)。
- 网络在训练的时候网络对训练集的cost在一直减小,然而网络对测试集的cost在下降一段时间后会保持上升随直到维持稳定(但这个过程中网络对测试集的分类精度还在不断的上升直到维持稳定)。
- 有时候网络对训练集的分类精度为100%然而对测试集的分类精度并没有达到100%,甚至只有80%多一点。这说明网络已经“记住”了测试集但泛化能力很差。
- 解决过拟合的一些手段
- 训练网络的过程中当网络对测试集的检测精度不再提升时应该尽快停止训练;
- 加大训练集样本的数量;
- 使用验证集监督过拟合问题:当网络在验证集上的检测精度不再上升时(保持稳定)停止网络的训练;
- regularization(正则化)
- 权值和权重的初始化方法也可以降低过拟合风险的概率
- 常见的方法
-
一般有L2 regularization,L2是在原来的损失函数C中添加一项【参考】以减小权值大小。这里给出了一些解释,其中一个观点是权值小则对噪声敏感度小从而使得噪声对其的影响也小。
-
L1 regularization,具体可以参考原文
上面的两个方法应该都可以成为weight decay
-
Dropout, modify the network itself. 在训练的过程中随机的删除与恢复部分神经元。一个简单的解释:我们使用多个网络实现同一个功能,最后综合所有网络的输出来获得最终的结果。这类似于投票,多个网络会降低过拟合概率。Dropout通过删除与恢复神经元使用类似投票的方法来降低过拟合。
-
Artificially expanding the training data(人为的数据增广),例如对图像进行旋转、对称变换、颜色亮度的更改等有效的手段增加训练集的大小
-
卷积神经网络【参考】
- 全连接一般比卷积层多40倍的参数量
- 基本概念【参考】
- Local receptive fields(感受野,或称为局部感知)
- Shared weights and biases(权值共享),对于同一个卷积核(或者称之为feature map、filter、kernels),后一层的神经元使用相同的权值(包括偏置值)。每一个卷积核都是一个滤波器,图像中有些地方响应度高有些地方低,这样就可以提取图像的空间信息。卷积核的参数就是感受野与后层神经元的权值(偏置值)集合。卷积核思想和haar-like思想很像。
- Pooling layers(池化层),池化层一般紧跟在卷积层之后。可以认为池化层是对信息的压缩,对于同一个结构,卷积核在其附近的响应应该都比较大,但我们可以只保留一个响应最大的点,同时降低了权值的个数,这是max-pooling的思想。L2 pooling:take the square root of the sum of the squares of the activations in the
x*x
region.(pooling使用的方法与图像尺寸压缩所使用的方法一致,对于2*2的pooling,pooling之后尺寸是原来的1/4)
- 说明参考原文
- 参考原文