弱者用泪水安慰自己,强者用汗水磨练自己。
上一篇文章里面讲了使用TensorFlow做手写数字图像识别,这篇文章算是它的进阶篇吧,在本篇文章中将会讲解如何使用TensorFlow识别多种类图片。本次使用的数据集是CIFAR-10,这是一个比较经典的数据集,可以去百度一下它的官网,它包含60000张32X32的彩色图像,其中训练集50000张,测试集10000张。里面一共是10类的图片,分别是airplane、automobile、bird、cat、deer、dog、frog、horse、ship和truck。
第一步我们需要下载TensorFlow Models库,你可以去github上面下载也可以使用git指令下载
git clone https://github.com/tensorflow/models.git
导入库,定义batch_size、训练轮数max_steps,以及下载CIFAR-10的路径
from tensorflow.models.tutorials.image.cifar10 import cifar10, cifar10_input
import tensorflow as tf
import numpy as np
import time
max_steps=3000
batch_size=128
data_dir='/cifar10_data'
定义初始化weight的函数,使用tf.truncated_normal截断的正太分布,给weight加一个L2的loss,L2正则化可以帮助我们筛选出最有效的特征。使用w1控制L2 loss的大小,使用tf.nn.l2_loss函数计算weight的L2 loss,再使用tf.multiply让L2 loss乘以w1,得到最后的weight loss,使用tf.add_to.collection把weight loss统一存到一个collection并命名为losses,以后计算神经网络总体的loss会用。
def variable_with_weight_loss(shape,stddev,w1):
var = tf.Variable(tf.truncated_normal(shape,stddev=stddev))
if w1 is not None:
weight_loss=tf.multiply(tf.nn.l2_loss(var),w1,name='weight_loss')
tf.add_to_collection('losses',weight_loss)
return var
使用cifar10来下载数据集,再使用cirfar10_input中的distorted_inputs函数产生训练需要使用的数据,包括特征及其对应的label,这里返回的是已经封装好的tensor,每次执行都会生成一个batch_size的数量的样本。里面使用了数据增强,包括随机的水平翻转、随机剪切一款24X24大小的图片、设置随机的亮度和对比度以及对数据进行标准化,如果你想了解更多,可以看看我之前写的文章,因为数据增强需要的计算量很大,所以该方法内部创建了16个独立的线程来进行工作,使用TensorFlow.queue进行调度
cifar10.maybe_download_and_extract()
images_train,labels_train=cifar10_input.distorted_inputs(data_dir=data_dir,batch_size=batch_size)
再使用cifar10_input.inputs来生成测试数据。创建holder,包含特征和label,因为batch_size在之后定义网络被用到了,所以数据尺寸中的第一个值需要被预先设定,大小为24X24,颜色通道为3。
images_test,labels_test=cifar10_input.inputs(eval_data=True,data_dir=data_dir,batch_size=batch_size)
image_holder=tf.placeholder(tf.float32,[batch_size,24,24,3])
label_holder=tf.placeholder(tf.int32,[batch_size])
开始创建第一个卷积层,先使用之前写好的variable_with_weight_loss函数创建卷积核的参数并初始化。第一个卷积层使用5X5的卷积核,3个颜色通道,64个卷积核,设置weight初始化参数的标准差为0.05。不对第一层卷积进行L2正则,所以w1设为0.使用tf.nn.conv2d函数对输入数据进行卷积操作,stride设为1,padding模式为SAME。把这层的bias全部初始化为0,再将卷积的结果加上bias,最后使用一个ReLU激活函数进行非线性化。在ReLU之后使用尺寸3X3,步长为2X2的最大池化层处理数据,然后使用tf.nn.lrn函数,该函数可以使反馈比较大的值更大,反馈比较小的值更小。
weight1=variable_with_weight_loss(shape=[5,5,3,64],stddev=5e-2,w1=0.0)
kernel1=tf.nn.conv2d(image_holder,weight1,[1,1,1,1],padding='SAME')
bias1=tf.Variable(tf.constant(0.0,shape=[64]))
conv1=tf.nn.relu(tf.nn.bias_add(kernel1,bias1))
pool1=tf.nn.max_pool(conv1,ksize=[1,3,3,1],strides=[1,2,2,1],padding='SAME')
norm1=tf.nn.lrn(pool1,4,bias1=1.0,alpha=0.01/9.0,beta=0.75)
第二层卷积步骤和第一层差不多,不同的是bias值全部初始化为0.1,最后再调换最大池化层和lrn层的位置。
weight2=variable_with_weight_loss(shape=[5,5,64,64],stddev=5e-2,w1=0.0)
kernel2=tf.nn.conv2d(norm1,weight2,[1,1,1,1],padding='SAME')
bias2=tf.Variable(tf.constant(0.1,shape=[64]))
conv2=tf.nn.relu(tf.nn.bias_add(kernel2,bias2))
norm2=tf.nn.lrn(conv2,4,bias=1.0,alpha=0.01/9.0,beta=0.75)
pool2=tf.nn.max_pool(norm2,ksize=[1,3,3,1],strides=[1,2,2,1],padding='SAME')
连接一个全连接层,将之前的输出结果flatten,使用tf.reshape函数将每个样本变成一维向量。使用get_shape获取数据扁平化后的长度。再使用variable_with_weight_loss函数对全连接层的weight进行初始化,这里的隐藏节点数为384,正太分布分标准差设为0.04,bias值初始化为0.1。需要注意的是我们不希望全连接层过拟合,所以设置了一个非零的weight loss值为0.04,让这一层所有的参数被L2正则约束。最后使用ReLU激活函数进行非线性化。
reshape=tf.reshape(pool2,[batch_size,-1])
dim=reshape.get_shape()[1].value
weight3=variable_with_weight_loss(shape=[dim,384],stddev=0.04,w1=0.004)
bias3=tf.Variable(tf.constant(0.1,shape=[384]))
local3=tf.nn.relu(tf.matmul(reshape,weight3)+bias3)
再来一层全连接,把隐藏节点数降低一半
weight4=variable_with_weight_loss(shape=[384,192],stddev=0.04,w1=0.004)
bias4=tf.Variable(tf.constant(0.1,shape=[192]))
local4=tf.nn.relu(tf.matmul(local3,weight4)+bias4)
创建最后一层,先创建weight,将其正太分布标准差设为上一隐含层的节点数的导数,并且不计入L2正则。
weight5=variable_with_weight_loss(shape=[192,10],stddev=1/192.0,w1=0.0)
bias5=tf.Variable(tf.constant(0.0,shape=[10]))
logits=tf.add(tf.matmul(local4,weight5),bias5)
接下来计算CNN的loss,计算softmax和cross_entropy_loss,使用tf.reduce_mean对cross_enteopy计算均值,再用tf.add_to_collection把cross_entropy的loss添加到整体losses的collection中。最后使用tf.add_n将全部loss求和
def loss(logits,labels):
labels=tf.cast(labels,tf.int64)
cross_entropy=tf.nn.sparse_softmax_cross_entropy_with_logits(logits=logits,labels=labels,name='cross_entropy_per_example')
cross_entropy_mean=tf.reduce_mean(cross_entropy,name='cross_entropy')
tf.add_to_collection('losses',cross_entropy_mean)
return tf.add_n(tf.get_collection('losses'),name='total_loss')
将logits节点和label_placeholder传入loss函数,得到最后的loss.
优化器使用adam,学习率设置为1e-3.
使用tf.nn.in_top_k函数求输出结果中top_k的准确率,默认使用top_1,也就是输出分数最高的那一类的准确率。
使用tf.InteractiveSession创建默认Session,初始化所有参数。
启动线程。
loss=loss(logits,label_holder)
train_op=tf.train.AdamOptimizer(1e-3).minimize(loss)
top_k_op=tf.nn.in_top_k(logits,label_holder,1)
sess=tf.InteractiveSession()
tf.global_variables_initializer().run()
tf.train.start_queue_runners()
开始正式训练,在每一个step的训练过程中,先用session的run方法执行image_train,labels_train的计算,获得一个batch的训练数据,再将这个batch的数据传入train_op和loss的计算。记录每一个step所消耗的时间,没10个step会打印一下loss,训练速率以及训练一个batch所消耗的时间。没有gpu会跑的比较慢。
for step in range(max_steps):
start_time=time.time()
image_batch,label_batch=sess.run([images_train,labels_train])
_,loss_value=sess.run([train_op,loss],feed_dict={image_holder:image_batch,label_holder:label_batch})
duration=time.time()-start_time
if step % 10==0:
examples_per_sec=batch_size/duration
sec_per_batch=float(duration)
format_str=('step %d,loss=%.2f(%.1f examples/sec; %.3f sec/batch)')
print(format_str%(step,loss_value,examples_per_sec,sec_per_batch))
接下来评测模型再测试集上的准确率,像训练那样一个batch一个batch进行测试,记录正确的数量,最后求得准确率并打印。
num_examples=10000
import math
num_iter=int(math.ceil(num_examples/batch_size))
true_count=0
total_sample_count=num_iter*batch_size
step=0
while step < num_iter:
image_batch,label_batch=sess.run([images_test,images_test])
predictions=sess.run([top_k_op],feed_dict={image_holder:image_batch,label_holder:label_batch})
true_count+=np.sum(predictions)
step+=1
precision=true_count/total_sample_count
print('precision @ 1=%.3f'%precision)