TensorFlow笔记(3)——利用TensorFlow和MNIST数据集训练一个最简单的手写数字识别模型

前言

当咱们开始学习编程的时候,第一件事每每是学习打印"Hello World"。就比如编程入门有Hello World,机器学习入门有MNIST。html

MNIST是一个入门级的计算机视觉数据集,它包含各类手写数字图片:python

手写数组

它也包含每一张图片对应的标签,告诉咱们这个是数字几。好比,上面这四张图片的标签分别是5,0,4,1web

其实训练一个简单的手写数字识别模型的代码很短,个人示例代码总共也就50行,除去注释、空格之类的估计连30行也没有,可是去理解包含在代码中的设计思想是很重要的,所以这篇笔记我会将我对每段代码的理解都记录下来。算法

参考:编程

MNIST机器学习入门api

机器学习-损失函数数组

MNIST数据集

MNIST数据集的官网是Yann LeCun's website。 虽然python提供了直接下载这个数据集的代码,可是考虑到国内网络的缘由,建议点这下载数据集,而后导入到项目根目录下就能够了。 bash

数据集
下载下来的数据集被分红两部分:60000行的训练数据集(mnist.train)和10000行的测试数据集(mnist.test)。这样的划分很重要,在机器学习模型设计时必须有一个单独的测试数据集不用于训练而是用来评估这个模型的性能,从而更加容易把设计的模型推广到其余数据集上(泛化)。

正如前面提到的同样,每个MNIST数据单元有两部分组成:一张包含手写数字的图片和一个对应的标签。咱们把这些图片设为xs,把这些标签设为ys。训练数据集和测试数据集都包含xsys,好比训练数据集的图片是 mnist.train.images,训练数据集的标签是 mnist.train.labels网络

每一张图片包含28像素X28像素。咱们能够用一个数字数组来表示这张图片:机器学习

示例

咱们把这个数组展开成一个向量,长度是 28x28 = 784。如何展开这个数组(数字间的顺序)不重要,只要保持各个图片采用相同的方式展开就能够了。

所以,在MNIST训练数据集中,mnist.train.images 是一个形状为 [60000, 784] 的张量,第一个维度数字用来索引图片,第二个维度数字用来索引每张图片中的像素点。在此张量里的每个元素,都表示某张图片里的某个像素的强度值,值介于0和1之间。

mnist.train.images

相对应的MNIST数据集的标签是介于0到9的数字,用来描述给定图片里表示的数字。为了用于这个教程,咱们使标签数据是"one-hot vectors"。 一个one-hot向量除了某一位的数字是1之外其他各维度数字都是0。因此在此教程中,数字n将表示成一个只有在第n维度(从0开始)数字为1的10维向量。好比,标签0将表示成([1,0,0,0,0,0,0,0,0,0,0])。所以, mnist.train.labels 是一个 [60000, 10] 的数字矩阵。

mnist.train.labels

如今,咱们准备好能够开始构建咱们的模型啦!

Softmax回归介绍

(由于这段很枯燥,并且我也解释不太好,因此干脆直接从Tensorflow的网站上复制粘贴来了,若是不想看的能够直接跳过到模型实现,最后写代码的时候只要知道用softmax函数就能够了)

咱们知道MNIST的每一张图片都表示一个数字,从0到9。咱们但愿获得给定图片表明每一个数字的几率。好比说,咱们的模型可能推测一张包含9的图片表明数字9的几率是80%可是判断它是8的几率是5%(由于8和9都有上半部分的小圆),而后给予它表明其余数字的几率更小的值。

这是一个使用softmax回归(softmax regression)模型的经典案例。softmax模型能够用来给不一样的对象分配几率。即便在以后,咱们训练更加精细的模型时,最后一步也须要用softmax来分配几率。

softmax回归(softmax regression)分两步:第一步

为了获得一张给定图片属于某个特定数字类的证据(evidence),咱们对图片像素值进行加权求和。若是这个像素具备很强的证听说明这张图片不属于该类,那么相应的权值为负数,相反若是这个像素拥有有利的证据支持这张图片属于这个类,那么权值是正数。

下面的图片显示了一个模型学习到的图片上每一个像素对于特定数字类的权值。红色表明负数权值,蓝色表明正数权值。

咱们也须要加入一个额外的偏置量(bias),由于输入每每会带有一些无关的干扰量。所以对于给定的输入图片 x 它表明的是数字 i 的证据能够表示为

其中 W_i 表明权重,b_i表明数字 i 类的偏置量,j 表明给定图片 x 的像素索引用于像素求和。而后用softmax函数能够把这些证据转换成几率 y

这里的softmax能够当作是一个激励(activation)函数或者连接(link)函数,把咱们定义的线性函数的输出转换成咱们想要的格式,也就是关于10个数字类的几率分布。所以,给定一张图片,它对于每个数字的吻合度能够被softmax函数转换成为一个几率值。softmax函数能够定义为:

展开等式右边的子式,能够获得:

可是更多的时候把softmax模型函数定义为前一种形式:把输入值当成幂指数求值,再正则化这些结果值。这个幂运算表示,更大的证据对应更大的假设模型(hypothesis)里面的乘数权重值。反之,拥有更少的证据意味着在假设模型里面拥有更小的乘数系数。假设模型里的权值不能够是0值或者负值。Softmax而后会正则化这些权重值,使它们的总和等于1,以此构造一个有效的几率分布。(更多的关于Softmax函数的信息,能够参考Michael Nieslen的书里面的这个部分,其中有关于softmax的可交互式的可视化解释。)

对于softmax回归模型能够用下面的图解释,对于输入的xs加权求和,再分别加上一个偏置量,最后再输入到softmax函数中:

若是把它写成一个等式,咱们能够获得:

咱们也能够用向量表示这个计算过程:用矩阵乘法和向量相加。这有助于提升计算效率。(也是一种更有效的思考方式)

更进一步,能够写成更加紧凑的方式:

实现模型

在使用TensorFlow以前,首先导入它:

import tensorflow as tf
复制代码

而后导入数据集并载入数据

from tensorflow.examples.tutorials.mnist import input_data

# 载入数据
mnist = input_data.read_data_sets("MNIST_data", one_hot=True)
复制代码

咱们经过操做符号变量来描述这些可交互的操做单元,能够用下面的方式建立一个:

x = tf.placeholder(tf.float32, [None, 784])
复制代码

x不是一个特定的值,而是一个占位符placeholder,咱们在TensorFlow运行计算时输入这个值。咱们但愿可以输入任意数量的MNIST图像,每一张图展平成784维的向量。咱们用2维的浮点数张量来表示这些图,这个张量的形状是[None,784 ]。(这里的None表示此张量的第一个维度能够是任何长度的。)

咱们的模型也须要权重和偏量,固然咱们能够把它们当作是另外的输入(使用占位符),但TensorFlow有一个更好的方法来表示它们:Variable 。 一个Variable表明一个可修改的张量,存在在TensorFlow的用于描述交互性操做的图中。它们能够用于计算输入值,也能够在计算中被修改。对于各类机器学习应用,通常都会有模型参数,能够用Variable表示。

W = tf.Variable(tf.zeros([784,10]))
b = tf.Variable(tf.zeros([10]))
复制代码

咱们赋予tf.Variable不一样的初值来建立不一样的Variable:在这里,咱们都用全为零的张量来初始化Wb。由于咱们要学习Wb的值,它们的初值能够随意设置。

注意,W的维度是[784,10],由于咱们想要用784维的图片向量乘以它以获得一个10维的证据值向量,每一位对应不一样数字类。b的形状是[10],因此咱们能够直接把它加到输出上面。

如今,咱们能够实现咱们的模型啦。只须要一行代码!

prediction = tf.nn.softmax(tf.matmul(x, W)+b)
复制代码

首先,咱们用tf.matmul(X,W)表示x乘以W,对应以前等式里面的Wx,这里x是一个2维张量拥有多个输入。而后再加上b,把和输入到tf.nn.softmax函数里面。

训练模型

为了训练咱们的模型,咱们首先须要定义一个指标来评估这个模型是好的。其实,在机器学习,咱们一般定义指标来表示一个模型是坏的,这个指标称为成本(cost)或损失(loss),而后尽可能最小化这个指标。可是,这两种方式是相同的。损失函数有不少种,在这里咱们采用平方损失函数,一般咱们会用均方差(MSE)做为衡量指标,公式以下:MSE=\frac{1}{n}\sum_{i=1}^{n}\ (y-prediction)^2

为了计算损失函数,咱们首先须要添加一个新的占位符用于输入正确值:

y = tf.placeholder(tf.float32, [None, 10])
复制代码

而后定义损失函数(loss):

# 二次代价函数
loss = tf.reduce_mean(tf.square(y-prediction))
复制代码

这段代码的含义我在上一篇笔记中已经介绍过了,不清楚的推荐阅读TensorFlow笔记(2)——利用TensorFlow训练一个最简单的一元线性模型

而后使用优化算法来不断的修改变量来下降损失值:

# 使用梯度降低法
train_step = tf.train.GradientDescentOptimizer(0.1).minimize(loss)
复制代码

在这里,咱们要求TensorFlow用梯度降低算法(gradient descent algorithm)以0.01的学习速率最小化交叉熵。梯度降低算法(gradient descent algorithm)是一个简单的学习过程,TensorFlow只需将每一个变量一点点地往使成本不断下降的方向移动。固然TensorFlow也提供了其余许多优化算法:只要简单地调整一行代码就可使用其余的算法。

TensorFlow在这里实际上所作的是,它会在后台给描述你的计算的那张图里面增长一系列新的计算操做单元用于实现反向传播算法和梯度降低算法。而后,它返回给你的只是一个单一的操做,当运行这个操做时,它用梯度降低算法训练你的模型,微调你的变量,不断减小成本。

如今,咱们已经设置好了咱们的模型。在运行计算以前,咱们须要添加一个操做来初始化咱们建立的变量:

# 初始化变量
init = tf.global_variables_initializer()
复制代码

接下来咱们就能够定义一个会话了,并在该会话中执行初始化变量的操做:

with tf.Session() as sess:
    sess.run(init)
复制代码

而后开始训练模型,咱们须要先定义一个批次batch_size,由于咱们在训练的时候不可能每次都只放一张图片进入神经网络(由于这样太慢了),批次为100在这表示的就是咱们一次放入100张图片进入神经网络,而后咱们须要计算一共会有多少个批次:

# 每一个批次的大小
batch_size = 100
# 计算一共有多少个批次
n_batch = mnist.train.num_examples // batch_size
复制代码

而后咱们让模型循环训练30次:

with tf.Session() as sess:
    sess.run(init)
    for epoch in range(30):
        for batch in range(n_batch):
            batch_xs, batch_ys = mnist.train.next_batch(batch_size)
            sess.run(train_step, feed_dict={x: batch_xs, y: batch_ys})
复制代码

该循环的每一个步骤中,咱们都会随机抓取训练数据中的n_batch个批处理数据点,而后咱们用这些数据点做为参数替换以前的占位符来运行train_step

使用一小部分的随机数据来进行训练被称为随机训练(stochastic training)- 在这里更确切的说是随机梯度降低训练。在理想状况下,咱们但愿用咱们全部的数据来进行每一步的训练,由于这能给咱们更好的训练结果,但显然这须要很大的计算开销。因此,每一次训练咱们可使用不一样的数据子集,这样作既能够减小计算开销,又能够最大化地学习到数据集的整体特性。

评估咱们的模型

那么咱们的模型性能如何呢?

首先让咱们找出那些预测正确的标签。tf.argmax 是一个很是有用的函数,它能给出某个tensor对象在某一维上的其数据最大值所在的索引值。因为标签向量是由0,1组成,所以最大值1所在的索引位置就是类别标签,好比tf.argmax(y,1)返回的是模型对于任一输入x预测到的标签值,而 tf.argmax(prediction,1) 表明正确的标签,咱们能够用 tf.equal 来检测咱们的预测是否真实标签匹配(索引位置同样表示匹配)。

# 结果存放在一个布尔型列表中
# argmax返回一维张量中最大的值所在的位置
correct_prediction = tf.equal(tf.argmax(y, 1), tf.argmax(prediction, 1))
复制代码

这行代码会给咱们一组布尔值。为了肯定正确预测项的比例,咱们能够把布尔值转换成浮点数,而后取平均值。例如,[True, False, True, True] 会变成 [1,0,1,1] ,取平均值后获得 0.75.

# 求准确率
accuracy = tf.reduce_mean(tf.cast(correct_prediction, tf.float32))
复制代码

最后,咱们计算所学习到的模型在测试数据集上面的正确率。

with tf.Session() as sess:
    sess.run(init)
    for epoch in range(30):
        for batch in range(n_batch):
            batch_xs, batch_ys = mnist.train.next_batch(batch_size)
            sess.run(train_step, feed_dict={x: batch_xs, y: batch_ys})
        acc = sess.run(accuracy, feed_dict={
                       x: mnist.test.images, y: mnist.test.labels})
        print("Iter "+str(epoch)+",Testing Accuracy "+str(acc))
复制代码

最终结果以下图所示,精确度大约在90%

image-20181204162301761

完整代码

我加了datetime这个包,目的是为了计算代码的执行时间,不影响阅读。

import datetime

# 3.2 MNIST数据集分类简单版本
import tensorflow as tf
from tensorflow.examples.tutorials.mnist import input_data

start = datetime.datetime.now()

# 载入数据
mnist = input_data.read_data_sets("MNIST_data", one_hot=True)
# 每一个批次的大小
batch_size = 100
# 计算一共有多少个批次
n_batch = mnist.train.num_examples // batch_size

# 定义两个placeholder
x = tf.placeholder(tf.float32, [None, 784])
y = tf.placeholder(tf.float32, [None, 10])

# 建立一个简单的神经网络
W = tf.Variable(tf.zeros([784, 10]))
b = tf.Variable(tf.zeros([10]))
prediction = tf.nn.softmax(tf.matmul(x, W)+b)

# 二次代价函数
loss = tf.reduce_mean(tf.square(y-prediction))
# 使用梯度降低法
train_step = tf.train.GradientDescentOptimizer(0.1).minimize(loss)

# 初始化变量
init = tf.global_variables_initializer()

# 结果存放在一个布尔型列表中
# argmax返回一维张量中最大的值所在的位置
correct_prediction = tf.equal(tf.argmax(y, 1), tf.argmax(prediction, 1))
# 求准确率
accuracy = tf.reduce_mean(tf.cast(correct_prediction, tf.float32))

with tf.Session() as sess:
    sess.run(init)
    for epoch in range(30):
        for batch in range(n_batch):
            batch_xs, batch_ys = mnist.train.next_batch(batch_size)
            sess.run(train_step, feed_dict={x: batch_xs, y: batch_ys})
        acc = sess.run(accuracy, feed_dict={
                       x: mnist.test.images, y: mnist.test.labels})
        print("Iter "+str(epoch)+",Testing Accuracy "+str(acc))

end = datetime.datetime.now()
print((end-start).seconds)
复制代码
相关文章
相关标签/搜索