赞
踩
内容都是百度AIstudio的内容,我只是在这里做个笔记,不是原创。
上一节我们研究了资源部署优化的方法,通过使用单GPU和分布式部署,提升模型训练的效率。本节我们依旧横向展开"横纵式",如 图1 所示,探讨在手写数字识别任务中,为了保证模型的真实效果,在模型训练部分,对模型进行一些调试和优化的方法。
训练过程优化思路主要有如下五个关键环节:
1. 计算分类准确率,观测模型训练效果。
交叉熵损失函数只能作为优化目标,无法直接准确衡量模型的训练效果。准确率可以直接衡量训练效果,但由于其离散性质,不适合做为损失函数优化神经网络。
2. 检查模型训练过程,识别潜在问题。
如果模型的损失或者评估指标表现异常,通常需要打印模型每一层的输入和输出来定位问题,分析每一层的内容来获取错误的原因。
3. 加入校验或测试,更好评价模型效果。
理想的模型训练结果是在训练集和验证集上均有较高的准确率,如果训练集上的准确率高于验证集,说明网络训练程度不够;如果验证集的准确率高于训练集,可能是发生了过拟合现象。通过在优化目标中加入正则化项的办法,解决过拟合的问题。
4. 加入正则化项,避免模型过拟合。
飞桨框架支持为整体参数加入正则化项,这是通常的做法。此外,飞桨框架也支持为某一层或某一部分的网络单独加入正则化项,以达到精细调整参数训练的效果。
5. 可视化分析。
用户不仅可以通过打印或使用matplotlib库作图,飞桨还集成了更专业的第三方绘图库tb-paddle,提供便捷的可视化分析。
- # 加载相关库
- import os
- import random
- import paddle
- import paddle.fluid as fluid
- from paddle.fluid.dygraph.nn import Conv2D, Pool2D, Linear
- import numpy as np
- from PIL import Image
-
- import gzip
- import json
-
- # 定义数据集读取器
- def load_data(mode='train'):
-
- # 读取数据文件
- datafile = './work/mnist.json.gz'
- print('loading mnist dataset from {} ......'.format(datafile))
- data = json.load(gzip.open(datafile))
- # 读取数据集中的训练集,验证集和测试集
- train_set, val_set, eval_set = data
-
- # 数据集相关参数,图片高度IMG_ROWS, 图片宽度IMG_COLS
- IMG_ROWS = 28
- IMG_COLS = 28
- # 根据输入mode参数决定使用训练集,验证集还是测试
- if mode == 'train':
- imgs = train_set[0]
- labels = train_set[1]
- elif mode == 'valid':
- imgs = val_set[0]
- labels = val_set[1]
- elif mode == 'eval':
- imgs = eval_set[0]
- labels = eval_set[1]
- # 获得所有图像的数量
- imgs_length = len(imgs)
- # 验证图像数量和标签数量是否一致
- assert len(imgs) == len(labels), \
- "length of train_imgs({}) should be the same as train_labels({})".format(
- len(imgs), len(labels))
-
- index_list = list(range(imgs_length))
-
- # 读入数据时用到的batchsize
- BATCHSIZE = 100
-
- # 定义数据生成器
- def data_generator():
- # 训练模式下,打乱训练数据
- if mode == 'train':
- random.shuffle(index_list)
- imgs_list = []
- labels_list = []
- # 按照索引读取数据
- for i in index_list:
- # 读取图像和标签,转换其尺寸和类型
- img = np.reshape(imgs[i], [1, IMG_ROWS, IMG_COLS]).astype('float32')
- label = np.reshape(labels[i], [1]).astype('int64')
- imgs_list.append(img)
- labels_list.append(label)
- # 如果当前数据缓存达到了batch size,就返回一个批次数据
- if len(imgs_list) == BATCHSIZE:
- yield np.array(imgs_list), np.array(labels_list)
- # 清空数据缓存列表
- imgs_list = []
- labels_list = []
-
- # 如果剩余数据的数目小于BATCHSIZE,
- # 则剩余数据一起构成一个大小为len(imgs_list)的mini-batch
- if len(imgs_list) > 0:
- yield np.array(imgs_list), np.array(labels_list)
-
- return data_generator
-
-
- # 定义模型结构
- class MNIST(fluid.dygraph.Layer):
- def __init__(self):
- super(MNIST, self).__init__()
-
- # 定义一个卷积层,使用relu激活函数
- self.conv1 = Conv2D(num_channels=1, num_filters=20, filter_size=5, stride=1, padding=2, act='relu')
- # 定义一个池化层,池化核为2,步长为2,使用最大池化方式
- self.pool1 = Pool2D(pool_size=2, pool_stride=2, pool_type='max')
- # 定义一个卷积层,使用relu激活函数
- self.conv2 = Conv2D(num_channels=20, num_filters=20, filter_size=5, stride=1, padding=2, act='relu')
- # 定义一个池化层,池化核为2,步长为2,使用最大池化方式
- self.pool2 = Pool2D(pool_size=2, pool_stride=2, pool_type='max')
- # 定义一个全连接层,输出节点数为10
- self.fc = Linear(input_dim=980, output_dim=10, act='softmax')
- # 定义网络的前向计算过程
- def forward(self, inputs, label=None):
- x = self.conv1(inputs)
- x = self.pool1(x)
- x = self.conv2(x)
- x = self.pool2(x)
- x = fluid.layers.reshape(x, [x.shape[0], 980])
- x = self.fc(x)
- if label is not None:
- acc = fluid.layers.accuracy(input=x, label=label)
- return x, acc
- else:
- return x
-
- #调用加载数据的函数
- train_loader = load_data('train')
-
- #在使用GPU机器时,可以将use_gpu变量设置成True
- use_gpu = False
- place = fluid.CUDAPlace(0) if use_gpu else fluid.CPUPlace()
-
- with fluid.dygraph.guard(place):
- model = MNIST()
- model.train()
-
- #四种优化算法的设置方案,可以逐一尝试效果
- optimizer = fluid.optimizer.SGDOptimizer(learning_rate=0.01, parameter_list=model.parameters())
- #optimizer = fluid.optimizer.MomentumOptimizer(learning_rate=0.01, momentum=0.9, parameter_list=model.parameters())
- #optimizer = fluid.optimizer.AdagradOptimizer(learning_rate=0.01, parameter_list=model.parameters())
- #optimizer = fluid.optimizer.AdamOptimizer(learning_rate=0.01, parameter_list=model.parameters())
-
- EPOCH_NUM = 5
- for epoch_id in range(EPOCH_NUM):
- for batch_id, data in enumerate(train_loader()):
- #准备数据
- image_data, label_data = data
- image = fluid.dygraph.to_variable(image_data)
- label = fluid.dygraph.to_variable(label_data)
-
- #前向计算的过程,同时拿到模型输出值和分类准确率
- predict, acc = model(image, label)
-
- #计算损失,取一个批次样本损失的平均值
- loss = fluid.layers.cross_entropy(predict, label)
- avg_loss = fluid.layers.mean(loss)
-
- #每训练了200批次的数据,打印下当前Loss的情况
- if batch_id % 200 == 0:
- print("epoch: {}, batch: {}, loss is: {}, acc is {}".format(epoch_id, batch_id, avg_loss.numpy(), acc.numpy()))
-
- #后向传播,更新参数的过程
- avg_loss.backward()
- optimizer.minimize(avg_loss)
- model.clear_gradients()
-
- #保存模型参数
- fluid.save_dygraph(model.state_dict(), 'mnist')
在训练过程中,我们会发现模型在训练样本集上的损失在不断减小。但这是否代表模型在未来的应用场景上依然有效?为了验证模型的有效性,通常将样本集合分成三份,训练集、校验集和测试集。
测试集上检验模型
代码1
- with fluid.dygraph.guard():
- print('start evaluation .......')
- #加载模型参数
- model = MNIST()
- model_state_dict, _ = fluid.load_dygraph('mnist')
- model.load_dict(model_state_dict)
-
- model.eval()
- eval_loader = load_data('eval')
-
- acc_set = []
- avg_loss_set = []
- cnt=0
- sum=0
- for batch_id, data in enumerate(eval_loader()):
- x_data, y_data = data
- img = fluid.dygraph.to_variable(x_data)
- label = fluid.dygraph.to_variable(y_data)
- prediction= model(img)
- # prediction=prediction.numpy().astype('int32')
- label=label.numpy().astype('int64')
- for i in range(len(label)):
- # print(prediction[i],label[i])
- tmp=np.argsort(prediction[i].numpy())
- if(tmp[-1]==label[i]):
- # print('aaaa:',tmp[-1],label[i])
- cnt+=1
- sum+=len(label)
- # print(len(prediction))
- # print("hello:",prediction.numpy().astype('int32'),label.numpy().astype('int32'))
- # loss = fluid.layers.square_error_cost(input=prediction, label=label)
- # avg_loss = fluid.layers.mean(loss)
- # acc_set.append(float(acc.numpy()))
- # avg_loss_set.append(float(avg_loss.numpy()))
-
- #计算多个batch的平均损失和准确率
- # acc_val_mean = np.array(acc_set).mean()
- # avg_loss_val_mean = np.array(avg_loss_set).mean()
-
- print("acc:",cnt/sum)
- # print('loss={}, acc={}'.format(avg_loss_val_mean, acc_val_mean))
代码2
- with fluid.dygraph.guard():
- print('start evaluation .......')
- #加载模型参数
- model = MNIST()
- model_state_dict, _ = fluid.load_dygraph('mnist')
- model.load_dict(model_state_dict)
-
- model.eval()
- eval_loader = load_data('eval')
-
- acc_set = []
- avg_loss_set = []
- for batch_id, data in enumerate(eval_loader()):
- x_data, y_data = data
- img = fluid.dygraph.to_variable(x_data)
- label = fluid.dygraph.to_variable(y_data)
- prediction, acc = model(img, label)
- loss = fluid.layers.cross_entropy(input=prediction, label=label)
- avg_loss = fluid.layers.mean(loss)
- acc_set.append(float(acc.numpy()))
- avg_loss_set.append(float(avg_loss.numpy()))
-
- #计算多个batch的平均损失和准确率
- acc_val_mean = np.array(acc_set).mean()
- avg_loss_val_mean = np.array(avg_loss_set).mean()
-
- print('loss={}, acc={}'.format(avg_loss_val_mean, acc_val_mean))
单张图片预测
- # 读取一张本地的样例图片,转变成模型输入的格式
- def load_image(img_path):
- # 从img_path中读取图像,并转为灰度图
- im = Image.open(img_path).convert('L')
- im.show()
- im = im.resize((28, 28), Image.ANTIALIAS)
- im = np.array(im).reshape(1, 1, 28, 28).astype(np.float32)
- # 图像归一化
- im = 1.0 - im / 127.5
- return im
-
- # 定义预测过程
- with fluid.dygraph.guard():
- model = MNIST()
- params_file_path = 'mnist'
- # img_path = './work/example_1.png'
- img_p= './work/example_'
- # 加载模型参数
- cnt=0
- model_dict, _ = fluid.load_dygraph("mnist")
- model.load_dict(model_dict)
-
- model.eval()
- for i in range(10):
- img_path=img_p+str(i)+'.png'
- tensor_img = load_image(img_path)
- #模型反馈10个分类标签的对应概率
- results = model(fluid.dygraph.to_variable(tensor_img))
- #取概率最大的标签作为预测输出
- lab = np.argsort(results.numpy())
- # print(lab)
- print("本次预测的数字是: ", lab[0][-1],i)
- if(lab[0][-1]==i):
- cnt+=1
- print(cnt/10)
对于样本量有限、但需要使用强大模型的复杂任务,模型很容易出现过拟合的表现,即在训练集上的损失小,在验证集或测试集上的损失较大,如 图2 所示。
反之,如果模型在训练集和测试集上均损失较大,则称为欠拟合。过拟合表示模型过于敏感,学习到了训练数据中的一些误差,而这些误差并不是真实的泛化规律(可推广到测试集上的规律)。欠拟合表示模型还不够强大,还没有很好的拟合已知的训练样本,更别提测试样本了。因为欠拟合情况容易观察和解决,只要训练loss不够好,就不断使用更强大的模型即可,因此实际中我们更需要处理好过拟合的问题。
造成过拟合的原因是模型过于敏感,而训练数据量太少或其中的噪音太多。
如图3 所示,理想的回归模型是一条坡度较缓的抛物线,欠拟合的模型只拟合出一条直线,显然没有捕捉到真实的规律,但过拟合的模型拟合出存在很多拐点的抛物线,显然是过于敏感,也没有正确表达真实规律。
如图4 所示,理想的分类模型是一条半圆形的曲线,欠拟合用直线作为分类边界,显然没有捕捉到真实的边界,但过拟合的模型拟合出很扭曲的分类边界,虽然对所有的训练数据正确分类,但对一些较为个例的样本所做出的妥协,高概率不是真实的规律。
归结到深度学习中,假设模型也会犯错,通过分析可能会由于如下两种情况导致:
情况1:训练数据存在噪音,导致模型学到了噪音,而不是真实规律。
情况2:使用强大模型(表示空间大)的同时训练数据太少,导致在训练数据上表现良好的候选假设太多,锁定了一个“虚假正确”的假设。
对于情况1,我们使用数据清洗和修正来解决。 对于情况2,我们或者限制模型表示能力,或者收集更多的训练数据。
而清洗训练数据中的错误,或收集更多的训练数据往往是一句“正确的废话”,在任何时候我们都想获得更多更高质量的数据。在实际项目中,更快、更低成本可控制过拟合的方法,只有限制模型的表示能力。
为了防止模型过拟合,在没有扩充样本量的可能下,只能降低模型的复杂度,可以通过限制参数的数量或可能取值(参数值尽量小)实现。
具体来说,在模型的优化目标(损失)中人为加入对参数规模的惩罚项。当参数越多或取值越大时,该惩罚项就越大。通过调整惩罚项的权重系数,可以使模型在“尽量减少训练损失”和“保持模型的泛化能力”之间取得平衡。泛化能力表示模型在没有见过的样本上依然有效。正则化项的存在,增加了模型在训练集上的损失。
飞桨框架支持为所有参数加上统一的正则化项,也支持为特定的参数添加正则化项。前者的实现如下代码所示,仅在优化器中设置regularization
参数即可实现。使用参数regularization_coeff
调节正则化项的权重,权重越大时,对模型复杂度的惩罚越高。实现代码如下所示。
- # 加载相关库
- import os
- import random
- import paddle
- import paddle.fluid as fluid
- from paddle.fluid.dygraph.nn import Conv2D, Pool2D, Linear
- import numpy as np
- from PIL import Image
-
- import gzip
- import json
-
- # 定义数据集读取器
- def load_data(mode='train'):
-
- # 读取数据文件
- datafile = './work/mnist.json.gz'
- print('loading mnist dataset from {} ......'.format(datafile))
- data = json.load(gzip.open(datafile))
- # 读取数据集中的训练集,验证集和测试集
- train_set, val_set, eval_set = data
-
- # 数据集相关参数,图片高度IMG_ROWS, 图片宽度IMG_COLS
- IMG_ROWS = 28
- IMG_COLS = 28
- # 根据输入mode参数决定使用训练集,验证集还是测试
- if mode == 'train':
- imgs = train_set[0]
- labels = train_set[1]
- elif mode == 'valid':
- imgs = val_set[0]
- labels = val_set[1]
- elif mode == 'eval':
- imgs = eval_set[0]
- labels = eval_set[1]
- # 获得所有图像的数量
- imgs_length = len(imgs)
- # 验证图像数量和标签数量是否一致
- assert len(imgs) == len(labels), \
- "length of train_imgs({}) should be the same as train_labels({})".format(
- len(imgs), len(labels))
-
- index_list = list(range(imgs_length))
-
- # 读入数据时用到的batchsize
- BATCHSIZE = 100
-
- # 定义数据生成器
- def data_generator():
- # 训练模式下,打乱训练数据
- if mode == 'train':
- random.shuffle(index_list)
- imgs_list = []
- labels_list = []
- # 按照索引读取数据
- for i in index_list:
- # 读取图像和标签,转换其尺寸和类型
- img = np.reshape(imgs[i], [1, IMG_ROWS, IMG_COLS]).astype('float32')
- label = np.reshape(labels[i], [1]).astype('int64')
- imgs_list.append(img)
- labels_list.append(label)
- # 如果当前数据缓存达到了batch size,就返回一个批次数据
- if len(imgs_list) == BATCHSIZE:
- yield np.array(imgs_list), np.array(labels_list)
- # 清空数据缓存列表
- imgs_list = []
- labels_list = []
-
- # 如果剩余数据的数目小于BATCHSIZE,
- # 则剩余数据一起构成一个大小为len(imgs_list)的mini-batch
- if len(imgs_list) > 0:
- yield np.array(imgs_list), np.array(labels_list)
-
- return data_generator
-
-
- # 定义模型结构
- class MNIST(fluid.dygraph.Layer):
- def __init__(self):
- super(MNIST, self).__init__()
-
- # 定义一个卷积层,使用relu激活函数
- self.conv1 = Conv2D(num_channels=1, num_filters=20, filter_size=5, stride=1, padding=2, act='relu')
- # 定义一个池化层,池化核为2,步长为2,使用最大池化方式
- self.pool1 = Pool2D(pool_size=2, pool_stride=2, pool_type='max')
- # 定义一个卷积层,使用relu激活函数
- self.conv2 = Conv2D(num_channels=20, num_filters=20, filter_size=5, stride=1, padding=2, act='relu')
- # 定义一个池化层,池化核为2,步长为2,使用最大池化方式
- self.pool2 = Pool2D(pool_size=2, pool_stride=2, pool_type='max')
- # 定义一个全连接层,输出节点数为10
- self.fc = Linear(input_dim=980, output_dim=10, act='softmax')
- # 定义网络的前向计算过程
- def forward(self, inputs, label=None):
- x = self.conv1(inputs)
- x = self.pool1(x)
- x = self.conv2(x)
- x = self.pool2(x)
- x = fluid.layers.reshape(x, [x.shape[0], 980])
- x = self.fc(x)
- if label is not None:
- acc = fluid.layers.accuracy(input=x, label=label)
- return x, acc
- else:
- return x
-
- #调用加载数据的函数
- train_loader = load_data('train')
-
- #在使用GPU机器时,可以将use_gpu变量设置成True
- use_gpu = True
- place = fluid.CUDAPlace(0) if use_gpu else fluid.CPUPlace()
-
- with fluid.dygraph.guard(place):
- model = MNIST()
- model.train()
-
- #四种优化算法的设置方案,可以逐一尝试效果
- optimizer = fluid.optimizer.SGDOptimizer(learning_rate=0.01, parameter_list=model.parameters())
- #optimizer = fluid.optimizer.MomentumOptimizer(learning_rate=0.01, momentum=0.9, parameter_list=model.parameters())
- #optimizer = fluid.optimizer.AdagradOptimizer(learning_rate=0.01, parameter_list=model.parameters())
- #optimizer = fluid.optimizer.AdamOptimizer(learning_rate=0.01, parameter_list=model.parameters())
-
- #各种优化算法均可以加入正则化项,避免过拟合,参数regularization_coeff调节正则化项的权重
- #optimizer = fluid.optimizer.SGDOptimizer(learning_rate=0.01, regularization=fluid.regularizer.L2Decay(regularization_coeff=0.1),parameter_list=model.parameters()))
- optimizer = fluid.optimizer.AdamOptimizer(learning_rate=0.01, regularization=fluid.regularizer.L2Decay(regularization_coeff=0.1),
- parameter_list=model.parameters())
-
- EPOCH_NUM = 10
- for epoch_id in range(EPOCH_NUM):
- for batch_id, data in enumerate(train_loader()):
- #准备数据,变得更加简洁
- image_data, label_data = data
- image = fluid.dygraph.to_variable(image_data)
- label = fluid.dygraph.to_variable(label_data)
-
- #前向计算的过程,同时拿到模型输出值和分类准确率
- predict, acc = model(image, label)
-
- #计算损失,取一个批次样本损失的平均值
- loss = fluid.layers.cross_entropy(predict, label)
- avg_loss = fluid.layers.mean(loss)
-
- #每训练了100批次的数据,打印下当前Loss的情况
- if batch_id % 100 == 0:
- print("epoch: {}, batch: {}, loss is: {}, acc is {}".format(epoch_id, batch_id, avg_loss.numpy(), acc.numpy()))
-
- #后向传播,更新参数的过程
- avg_loss.backward()
- optimizer.minimize(avg_loss)
- model.clear_gradients()
-
- #保存模型参数
- fluid.save_dygraph(model.state_dict(), 'mnist')
训练模型时,经常需要观察模型的评价指标,分析模型的优化过程,以确保训练是有效的。可视化分析有两种工具:Matplotlib库和tb-paddle。
使用Matplotlib库绘制损失随训练下降的曲线图
- # 加载相关库
- import os
- import random
- import paddle
- import paddle.fluid as fluid
- from paddle.fluid.dygraph.nn import Conv2D, Pool2D, Linear
- #引入matplotlib库
- import matplotlib.pyplot as plt
- import numpy as np
- from PIL import Image
-
- import gzip
- import json
-
- # 定义数据集读取器
- def load_data(mode='train'):
-
- # 读取数据文件
- datafile = './work/mnist.json.gz'
- print('loading mnist dataset from {} ......'.format(datafile))
- data = json.load(gzip.open(datafile))
- # 读取数据集中的训练集,验证集和测试集
- train_set, val_set, eval_set = data
-
- # 数据集相关参数,图片高度IMG_ROWS, 图片宽度IMG_COLS
- IMG_ROWS = 28
- IMG_COLS = 28
- # 根据输入mode参数决定使用训练集,验证集还是测试
- if mode == 'train':
- imgs = train_set[0]
- labels = train_set[1]
- elif mode == 'valid':
- imgs = val_set[0]
- labels = val_set[1]
- elif mode == 'eval':
- imgs = eval_set[0]
- labels = eval_set[1]
- # 获得所有图像的数量
- imgs_length = len(imgs)
- # 验证图像数量和标签数量是否一致
- assert len(imgs) == len(labels), \
- "length of train_imgs({}) should be the same as train_labels({})".format(
- len(imgs), len(labels))
-
- index_list = list(range(imgs_length))
-
- # 读入数据时用到的batchsize
- BATCHSIZE = 100
-
- # 定义数据生成器
- def data_generator():
- # 训练模式下,打乱训练数据
- if mode == 'train':
- random.shuffle(index_list)
- imgs_list = []
- labels_list = []
- # 按照索引读取数据
- for i in index_list:
- # 读取图像和标签,转换其尺寸和类型
- img = np.reshape(imgs[i], [1, IMG_ROWS, IMG_COLS]).astype('float32')
- label = np.reshape(labels[i], [1]).astype('int64')
- imgs_list.append(img)
- labels_list.append(label)
- # 如果当前数据缓存达到了batch size,就返回一个批次数据
- if len(imgs_list) == BATCHSIZE:
- yield np.array(imgs_list), np.array(labels_list)
- # 清空数据缓存列表
- imgs_list = []
- labels_list = []
-
- # 如果剩余数据的数目小于BATCHSIZE,
- # 则剩余数据一起构成一个大小为len(imgs_list)的mini-batch
- if len(imgs_list) > 0:
- yield np.array(imgs_list), np.array(labels_list)
-
- return data_generator
-
-
- # 定义模型结构
- class MNIST(fluid.dygraph.Layer):
- def __init__(self):
- super(MNIST, self).__init__()
-
- # 定义一个卷积层,使用relu激活函数
- self.conv1 = Conv2D(num_channels=1, num_filters=20, filter_size=5, stride=1, padding=2, act='relu')
- # 定义一个池化层,池化核为2,步长为2,使用最大池化方式
- self.pool1 = Pool2D(pool_size=2, pool_stride=2, pool_type='max')
- # 定义一个卷积层,使用relu激活函数
- self.conv2 = Conv2D(num_channels=20, num_filters=20, filter_size=5, stride=1, padding=2, act='relu')
- # 定义一个池化层,池化核为2,步长为2,使用最大池化方式
- self.pool2 = Pool2D(pool_size=2, pool_stride=2, pool_type='max')
- # 定义一个全连接层,输出节点数为10
- self.fc = Linear(input_dim=980, output_dim=10, act='softmax')
- # 定义网络的前向计算过程
- def forward(self, inputs, label=None):
- x = self.conv1(inputs)
- x = self.pool1(x)
- x = self.conv2(x)
- x = self.pool2(x)
- x = fluid.layers.reshape(x, [x.shape[0], 980])
- x = self.fc(x)
- if label is not None:
- acc = fluid.layers.accuracy(input=x, label=label)
- return x, acc
- else:
- return x
-
- #调用加载数据的函数
- train_loader = load_data('train')
-
- #在使用GPU机器时,可以将use_gpu变量设置成True
- use_gpu = True
- place = fluid.CUDAPlace(0) if use_gpu else fluid.CPUPlace()
-
- with fluid.dygraph.guard(place):
- model = MNIST()
- model.train()
-
- #四种优化算法的设置方案,可以逐一尝试效果
- optimizer = fluid.optimizer.SGDOptimizer(learning_rate=0.01, parameter_list=model.parameters())
-
- EPOCH_NUM = 10
- iter=0
- iters=[]
- losses=[]
- for epoch_id in range(EPOCH_NUM):
- for batch_id, data in enumerate(train_loader()):
- #准备数据,变得更加简洁
- image_data, label_data = data
- image = fluid.dygraph.to_variable(image_data)
- label = fluid.dygraph.to_variable(label_data)
-
- #前向计算的过程,同时拿到模型输出值和分类准确率
- predict, acc = model(image, label)
-
- #计算损失,取一个批次样本损失的平均值
- loss = fluid.layers.cross_entropy(predict, label)
- avg_loss = fluid.layers.mean(loss)
-
- #每训练了100批次的数据,打印下当前Loss的情况
- if batch_id % 100 == 0:
- print("epoch: {}, batch: {}, loss is: {}, acc is {}".format(epoch_id, batch_id, avg_loss.numpy(), acc.numpy()))
- iters.append(iter)
- losses.append(avg_loss.numpy())
- iter = iter + 100
-
- #后向传播,更新参数的过程
- avg_loss.backward()
- optimizer.minimize(avg_loss)
- model.clear_gradients()
-
- #保存模型参数
- fluid.save_dygraph(model.state_dict(), 'mnist')
- #画出训练过程中Loss的变化曲线
- plt.figure()
- plt.title("train loss", fontsize=24)
- plt.xlabel("iter", fontsize=14)
- plt.ylabel("loss", fontsize=14)
- plt.plot(iters, losses,color='red',label='train loss')
- plt.grid()
- plt.show()
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。