图像分类是根据图像的语义信息对不同类别图像进行区分,是计算机视觉中重要的基础问题,是物体检测、图像分割、物体跟踪、行为分析、人脸识别等其他高层次视觉任务的基础。图像分类在许多领域都有着广泛的应用,如:安防领域的人脸识别和智能视频分析等,交通领域的交通场景识别,互联网领域基于内容的图像检索和相册自动归类,医学领域的图像识别等。

上一节主要介绍了卷积神经网络常用的一些基本模块,本节将基于眼疾分类数据集iChallenge-PM,对图像分类领域的经典卷积神经网络进行剖析,介绍如何应用这些基础模块构建卷积神经网络,解决图像分类问题。涵盖如下卷积神经网络:

  • LeNet:Yan LeCun等人于1998年第一次将卷积神经网络应用到图像分类任务上[1],在手写数字识别任务上取得了巨大成功。

  • AlexNet:Alex Krizhevsky等人在2012年提出了AlexNet[2], 并应用在大尺寸图片数据集ImageNet上,获得了2012年ImageNet比赛冠军(ImageNet Large Scale Visual Recognition Challenge,ILSVRC)。

  • VGG:Simonyan和Zisserman于2014年提出了VGG网络结构[3],是当前最流行的卷积神经网络之一,由于其结构简单、应用性极强而深受广受研究者欢迎。

  • GoogLeNet:Christian Szegedy等人在2014提出了GoogLeNet[4],并取得了2014年ImageNet比赛冠军。

  • ResNet:Kaiming He等人在2015年提出了ResNet[5],通过引入残差模块加深网络层数,在ImagNet数据集上的识别错误率降低到3.6%,超越了人眼识别水平。ResNet的设计思想深刻的影响了后来的深度神经网络的设计。

LeNet

LeNet是最早的卷积神经网络之一[1]。1998年,Yan LeCun第一次将LeNet卷积神经网络应用到图像分类上,在手写数字识别任务中取得了巨大成功。LeNet通过连续使用卷积和池化层的组合提取图像特征,其架构如 图1 所示,这里展示的是作者论文中的LeNet-5模型:

图像分类 - 图1

图1:LeNet模型网络结构示意图

  • 第一轮卷积和池化:卷积提取图像中包含的特征模式(激活函数使用sigmoid),图像尺寸从32减小到28。经过池化层可以降低输出特征图对空间位置的敏感性,图像尺寸减到14。

  • 第二轮卷积和池化:卷积操作使图像尺寸减小到10,经过池化后变成5。

  • 第三轮卷积:将经过第3次卷积提取到的特征图输入到全连接层。第一个全连接层的输出神经元的个数是64,第二个全连接层的输出神经元个数是分类标签的类别数,对于手写数字识别其大小是10。然后使用Softmax激活函数即可计算出每个类别的预测概率。


【提示】:

卷积层的输出特征图如何当作全连接层的输入使用呢?

卷积层的输出数据格式是

图像分类 - 图2 ,在输入全连接层的时候,会自动将数据拉平,

也就是对每个样本,自动将其转化为长度为

图像分类 - 图3 的向量,

其中

图像分类 - 图4 ,一个mini-batch的数据维度变成了 图像分类 - 图5 的二维向量。


LeNet在手写数字识别上的应用

LeNet网络的实现代码如下:

  1. # 导入需要的包
  2. import paddle
  3. import paddle.fluid as fluid
  4. import numpy as np
  5. from paddle.fluid.dygraph.nn import Conv2D, Pool2D, Linear
  6. # 定义 LeNet 网络结构
  7. class LeNet(fluid.dygraph.Layer):
  8. def __init__(self, name_scope, num_classes=1):
  9. super(LeNet, self).__init__(name_scope)
  10. # 创建卷积和池化层块,每个卷积层使用Sigmoid激活函数,后面跟着一个2x2的池化
  11. self.conv1 = Conv2D(num_channels=1, num_filters=6, filter_size=5, act='sigmoid')
  12. self.pool1 = Pool2D(pool_size=2, pool_stride=2, pool_type='max')
  13. self.conv2 = Conv2D(num_channels=6, num_filters=16, filter_size=5, act='sigmoid')
  14. self.pool2 = Pool2D(pool_size=2, pool_stride=2, pool_type='max')
  15. # 创建第3个卷积层
  16. self.conv3 = Conv2D(num_channels=16, num_filters=120, filter_size=4, act='sigmoid')
  17. # 创建全连接层,第一个全连接层的输出神经元个数为64, 第二个全连接层输出神经元个数为分裂标签的类别数
  18. self.fc1 = Linear(input_dim=120, output_dim=64, act='sigmoid')
  19. self.fc2 = Linear(input_dim=64, output_dim=num_classes)
  20. # 网络的前向计算过程
  21. def forward(self, x):
  22. x = self.conv1(x)
  23. x = self.pool1(x)
  24. x = self.conv2(x)
  25. x = self.pool2(x)
  26. x = self.conv3(x)
  27. x = fluid.layers.reshape(x, [x.shape[0], -1])
  28. x = self.fc1(x)
  29. x = self.fc2(x)
  30. return x

下面的程序使用随机数作为输入,查看经过LeNet-5的每一层作用之后,输出数据的形状

  1. # 输入数据形状是 [N, 1, H, W]
  2. # 这里用np.random创建一个随机数组作为输入数据
  3. x = np.random.randn(*[3,1,28,28])
  4. x = x.astype('float32')
  5. with fluid.dygraph.guard():
  6. # 创建LeNet类的实例,指定模型名称和分类的类别数目
  7. m = LeNet('LeNet', num_classes=10)
  8. # 通过调用LeNet从基类继承的sublayers()函数,
  9. # 查看LeNet中所包含的子层
  10. print(m.sublayers())
  11. x = fluid.dygraph.to_variable(x)
  12. for item in m.sublayers():
  13. # item是LeNet类中的一个子层
  14. # 查看经过子层之后的输出数据形状
  15. try:
  16. x = item(x)
  17. except:
  18. x = fluid.layers.reshape(x, [x.shape[0], -1])
  19. x = item(x)
  20. if len(item.parameters())==2:
  21. # 查看卷积和全连接层的数据和参数的形状,
  22. # 其中item.parameters()[0]是权重参数w,item.parameters()[1]是偏置参数b
  23. print(item.full_name(), x.shape, item.parameters()[0].shape, item.parameters()[1].shape)
  24. else:
  25. # 池化层没有参数
  26. print(item.full_name(), x.shape)
  1. [<paddle.fluid.dygraph.nn.Conv2D object at 0x7f29858ebad0>, <paddle.fluid.dygraph.nn.Pool2D object at 0x7f29858f8110>, <paddle.fluid.dygraph.nn.Conv2D object at 0x7f29858f8230>, <paddle.fluid.dygraph.nn.Pool2D object at 0x7f29858f82f0>, <paddle.fluid.dygraph.nn.Conv2D object at 0x7f29858f8350>, <paddle.fluid.dygraph.nn.Linear object at 0x7f29858f8470>, <paddle.fluid.dygraph.nn.Linear object at 0x7f29858f85f0>]
  2. conv2d_0 [3, 6, 24, 24] [6, 1, 5, 5] [6]
  3. pool2d_0 [3, 6, 12, 12]
  4. conv2d_1 [3, 16, 8, 8] [16, 6, 5, 5] [16]
  5. pool2d_1 [3, 16, 4, 4]
  6. conv2d_2 [3, 120, 1, 1] [120, 16, 4, 4] [120]
  7. linear_0 [3, 64] [120, 64] [64]
  8. linear_1 [3, 10] [64, 10] [10]
  1. # -*- coding: utf-8 -*-
  2. # LeNet 识别手写数字
  3. import os
  4. import random
  5. import paddle
  6. import paddle.fluid as fluid
  7. import numpy as np
  8. # 定义训练过程
  9. def train(model):
  10. print('start training ... ')
  11. model.train()
  12. epoch_num = 5
  13. opt = fluid.optimizer.Momentum(learning_rate=0.001, momentum=0.9, parameter_list=model.parameters())
  14. # 使用Paddle自带的数据读取器
  15. train_loader = paddle.batch(paddle.dataset.mnist.train(), batch_size=10)
  16. valid_loader = paddle.batch(paddle.dataset.mnist.test(), batch_size=10)
  17. for epoch in range(epoch_num):
  18. for batch_id, data in enumerate(train_loader()):
  19. # 调整输入数据形状和类型
  20. x_data = np.array([item[0] for item in data], dtype='float32').reshape(-1, 1, 28, 28)
  21. y_data = np.array([item[1] for item in data], dtype='int64').reshape(-1, 1)
  22. # 将numpy.ndarray转化成Tensor
  23. img = fluid.dygraph.to_variable(x_data)
  24. label = fluid.dygraph.to_variable(y_data)
  25. # 计算模型输出
  26. logits = model(img)
  27. # 计算损失函数
  28. loss = fluid.layers.softmax_with_cross_entropy(logits, label)
  29. avg_loss = fluid.layers.mean(loss)
  30. if batch_id % 1000 == 0:
  31. print("epoch: {}, batch_id: {}, loss is: {}".format(epoch, batch_id, avg_loss.numpy()))
  32. avg_loss.backward()
  33. opt.minimize(avg_loss)
  34. model.clear_gradients()
  35. model.eval()
  36. accuracies = []
  37. losses = []
  38. for batch_id, data in enumerate(valid_loader()):
  39. # 调整输入数据形状和类型
  40. x_data = np.array([item[0] for item in data], dtype='float32').reshape(-1, 1, 28, 28)
  41. y_data = np.array([item[1] for item in data], dtype='int64').reshape(-1, 1)
  42. # 将numpy.ndarray转化成Tensor
  43. img = fluid.dygraph.to_variable(x_data)
  44. label = fluid.dygraph.to_variable(y_data)
  45. # 计算模型输出
  46. logits = model(img)
  47. pred = fluid.layers.softmax(logits)
  48. # 计算损失函数
  49. loss = fluid.layers.softmax_with_cross_entropy(logits, label)
  50. acc = fluid.layers.accuracy(pred, label)
  51. accuracies.append(acc.numpy())
  52. losses.append(loss.numpy())
  53. print("[validation] accuracy/loss: {}/{}".format(np.mean(accuracies), np.mean(losses)))
  54. model.train()
  55. # 保存模型参数
  56. fluid.save_dygraph(model.state_dict(), 'mnist')
  57. if __name__ == '__main__':
  58. # 创建模型
  59. with fluid.dygraph.guard():
  60. model = LeNet("LeNet", num_classes=10)
  61. #启动训练过程
  62. train(model)
  1. start training ...
  1. Cache file /home/aistudio/.cache/paddle/dataset/mnist/train-images-idx3-ubyte.gz not found, downloading https://dataset.bj.bcebos.com/mnist/train-images-idx3-ubyte.gz
  2. Begin to download
  3.  
  4. Download finished
  5. Cache file /home/aistudio/.cache/paddle/dataset/mnist/train-labels-idx1-ubyte.gz not found, downloading https://dataset.bj.bcebos.com/mnist/train-labels-idx1-ubyte.gz
  6. Begin to download
  7. ........
  8. Download finished
  9. Cache file /home/aistudio/.cache/paddle/dataset/mnist/t10k-images-idx3-ubyte.gz not found, downloading https://dataset.bj.bcebos.com/mnist/t10k-images-idx3-ubyte.gz
  10. Begin to download
  11.  
  12. Download finished
  13. Cache file /home/aistudio/.cache/paddle/dataset/mnist/t10k-labels-idx1-ubyte.gz not found, downloading https://dataset.bj.bcebos.com/mnist/t10k-labels-idx1-ubyte.gz
  14. Begin to download
  15. ..
  16. Download finished
  1. epoch: 0, batch_id: 0, loss is: [2.5567963]
  2. epoch: 0, batch_id: 1000, loss is: [2.2921207]
  3. epoch: 0, batch_id: 2000, loss is: [2.329089]
  4. epoch: 0, batch_id: 3000, loss is: [2.2760074]
  5. epoch: 0, batch_id: 4000, loss is: [2.2555802]
  6. epoch: 0, batch_id: 5000, loss is: [2.321007]
  7. [validation] accuracy/loss: 0.3555000126361847/2.2462358474731445
  8. epoch: 1, batch_id: 0, loss is: [2.2364979]
  9. epoch: 1, batch_id: 1000, loss is: [2.1558306]
  10. epoch: 1, batch_id: 2000, loss is: [2.1844604]
  11. epoch: 1, batch_id: 3000, loss is: [1.7957464]
  12. epoch: 1, batch_id: 4000, loss is: [1.341808]
  13. epoch: 1, batch_id: 5000, loss is: [1.6028554]
  14. [validation] accuracy/loss: 0.7293000221252441/1.0572129487991333
  15. epoch: 2, batch_id: 0, loss is: [0.85837966]
  16. epoch: 2, batch_id: 1000, loss is: [0.6425297]
  17. epoch: 2, batch_id: 2000, loss is: [0.6375253]
  18. epoch: 2, batch_id: 3000, loss is: [0.40348434]
  19. epoch: 2, batch_id: 4000, loss is: [0.37101394]
  20. epoch: 2, batch_id: 5000, loss is: [0.65031445]
  21. [validation] accuracy/loss: 0.8730000257492065/0.47411048412323
  22. epoch: 3, batch_id: 0, loss is: [0.35694075]
  23. epoch: 3, batch_id: 1000, loss is: [0.25489596]
  24. epoch: 3, batch_id: 2000, loss is: [0.29641074]
  25. epoch: 3, batch_id: 3000, loss is: [0.18106733]
  26. epoch: 3, batch_id: 4000, loss is: [0.1899938]
  27. epoch: 3, batch_id: 5000, loss is: [0.32796213]
  28. [validation] accuracy/loss: 0.9122999906539917/0.3133768141269684
  29. epoch: 4, batch_id: 0, loss is: [0.24354395]
  30. epoch: 4, batch_id: 1000, loss is: [0.16107734]
  31. epoch: 4, batch_id: 2000, loss is: [0.20161033]
  32. epoch: 4, batch_id: 3000, loss is: [0.09298491]
  33. epoch: 4, batch_id: 4000, loss is: [0.11935985]
  34. epoch: 4, batch_id: 5000, loss is: [0.19827338]
  35. [validation] accuracy/loss: 0.9312999844551086/0.23992861807346344

通过运行结果可以看出,LeNet在手写数字识别MNIST验证数据集上的准确率高达92%以上。那么对于其它数据集效果如何呢?我们通过眼疾识别数据集iChallenge-PM验证一下。

LeNet在眼疾识别数据集iChallenge-PM上的应用

iChallenge-PM是百度大脑和中山大学中山眼科中心联合举办的iChallenge比赛中,提供的关于病理性近视(Pathologic Myopia,PM)的医疗类数据集,包含1200个受试者的眼底视网膜图片,训练、验证和测试数据集各400张。下面我们详细介绍LeNet在iChallenge-PM上的训练过程。


说明:

如今近视已经成为困扰人们健康的一项全球性负担,在近视人群中,有超过35%的人患有重度近视。近似将会导致眼睛的光轴被拉长,有可能引起视网膜或者络网膜的病变。随着近似度数的不断加深,高度近似有可能引发病理性病变,这将会导致以下几种症状:视网膜或者络网膜发生退化、视盘区域萎缩、漆裂样纹损害、Fuchs斑等。因此,及早发现近似患者眼睛的病变并采取治疗,显得非常重要。

数据可以从AIStudio下载


数据集准备

/home/aistudio/data/data19065 目录包括如下三个文件,解压缩后存放在/home/aistudio/work/palm目录下。

  • training.zip:包含训练中的图片和标签
  • validation.zip:包含验证集的图片
  • valid_gt.zip:包含验证集的标签

注意

valid_gt.zip文件解压缩之后,需要将/home/aistudio/work/palm/PALM-Validation-GT/目录下的PM_Label_and_Fovea_Location.xlsx文件转存成csv格式,本节代码示例中已经提前转成文件labels.csv。


  1. # 初次运行时将注释取消,以便解压文件
  2. # 如果已经解压过了,则不需要运行此段代码,否则文件已经存在解压会报错
  3. !unzip -o -q -d /home/aistudio/work/palm /home/aistudio/data/data19065/training.zip
  4. %cd /home/aistudio/work/palm/PALM-Training400/
  5. !unzip -o -q PALM-Training400.zip
  6. !unzip -o -q -d /home/aistudio/work/palm /home/aistudio/data/data19065/validation.zip
  7. !unzip -o -q -d /home/aistudio/work/palm /home/aistudio/data/data19065/valid_gt.zip
  1. /home/aistudio/work/palm/PALM-Training400

查看数据集图片

iChallenge-PM中既有病理性近视患者的眼底图片,也有非病理性近视患者的图片,命名规则如下:

  • 病理性近视(PM):文件名以P开头

  • 非病理性近视(non-PM):

    • 高度近似(high myopia):文件名以H开头

    • 正常眼睛(normal):文件名以N开头

我们将病理性患者的图片作为正样本,标签为1; 非病理性患者的图片作为负样本,标签为0。从数据集中选取两张图片,通过LeNet提取特征,构建分类器,对正负样本进行分类,并将图片显示出来。代码如下所示:

  1. import os
  2. import numpy as np
  3. import matplotlib.pyplot as plt
  4. %matplotlib inline
  5. from PIL import Image
  6. DATADIR = '/home/aistudio/work/palm/PALM-Training400/PALM-Training400'
  7. # 文件名以N开头的是正常眼底图片,以P开头的是病变眼底图片
  8. file1 = 'N0012.jpg'
  9. file2 = 'P0095.jpg'
  10. # 读取图片
  11. img1 = Image.open(os.path.join(DATADIR, file1))
  12. img1 = np.array(img1)
  13. img2 = Image.open(os.path.join(DATADIR, file2))
  14. img2 = np.array(img2)
  15. # 画出读取的图片
  16. plt.figure(figsize=(16, 8))
  17. f = plt.subplot(121)
  18. f.set_title('Normal', fontsize=20)
  19. plt.imshow(img1)
  20. f = plt.subplot(122)
  21. f.set_title('PM', fontsize=20)
  22. plt.imshow(img2)
  23. plt.show()
  1. 2020-03-25 19:44:41,518-INFO: font search path ['/opt/conda/envs/python35-paddle120-env/lib/python3.7/site-packages/matplotlib/mpl-data/fonts/ttf', '/opt/conda/envs/python35-paddle120-env/lib/python3.7/site-packages/matplotlib/mpl-data/fonts/afm', '/opt/conda/envs/python35-paddle120-env/lib/python3.7/site-packages/matplotlib/mpl-data/fonts/pdfcorefonts']
  2. 2020-03-25 19:44:41,916-INFO: generated new fontManager

图像分类 - 图6

  1. <Figure size 1152x576 with 2 Axes>
  1. # 查看图片形状
  2. img1.shape, img2.shape
  1. ((2056, 2124, 3), (2056, 2124, 3))

定义数据读取器

使用OpenCV从磁盘读入图片,将每张图缩放到

图像分类 - 图7 大小,并且将像素值调整到 图像分类 - 图8 之间,代码如下所示:

  1. import cv2
  2. import random
  3. import numpy as np
  4. # 对读入的图像数据进行预处理
  5. def transform_img(img):
  6. # 将图片尺寸缩放道 224x224
  7. img = cv2.resize(img, (224, 224))
  8. # 读入的图像数据格式是[H, W, C]
  9. # 使用转置操作将其变成[C, H, W]
  10. img = np.transpose(img, (2,0,1))
  11. img = img.astype('float32')
  12. # 将数据范围调整到[-1.0, 1.0]之间
  13. img = img / 255.
  14. img = img * 2.0 - 1.0
  15. return img
  16. # 定义训练集数据读取器
  17. def data_loader(datadir, batch_size=10, mode = 'train'):
  18. # 将datadir目录下的文件列出来,每条文件都要读入
  19. filenames = os.listdir(datadir)
  20. def reader():
  21. if mode == 'train':
  22. # 训练时随机打乱数据顺序
  23. random.shuffle(filenames)
  24. batch_imgs = []
  25. batch_labels = []
  26. for name in filenames:
  27. filepath = os.path.join(datadir, name)
  28. img = cv2.imread(filepath)
  29. img = transform_img(img)
  30. if name[0] == 'H' or name[0] == 'N':
  31. # H开头的文件名表示高度近似,N开头的文件名表示正常视力
  32. # 高度近视和正常视力的样本,都不是病理性的,属于负样本,标签为0
  33. label = 0
  34. elif name[0] == 'P':
  35. # P开头的是病理性近视,属于正样本,标签为1
  36. label = 1
  37. else:
  38. raise('Not excepted file name')
  39. # 每读取一个样本的数据,就将其放入数据列表中
  40. batch_imgs.append(img)
  41. batch_labels.append(label)
  42. if len(batch_imgs) == batch_size:
  43. # 当数据列表的长度等于batch_size的时候,
  44. # 把这些数据当作一个mini-batch,并作为数据生成器的一个输出
  45. imgs_array = np.array(batch_imgs).astype('float32')
  46. labels_array = np.array(batch_labels).astype('float32').reshape(-1, 1)
  47. yield imgs_array, labels_array
  48. batch_imgs = []
  49. batch_labels = []
  50. if len(batch_imgs) > 0:
  51. # 剩余样本数目不足一个batch_size的数据,一起打包成一个mini-batch
  52. imgs_array = np.array(batch_imgs).astype('float32')
  53. labels_array = np.array(batch_labels).astype('float32').reshape(-1, 1)
  54. yield imgs_array, labels_array
  55. return reader
  56. # 定义验证集数据读取器
  57. def valid_data_loader(datadir, csvfile, batch_size=10, mode='valid'):
  58. # 训练集读取时通过文件名来确定样本标签,验证集则通过csvfile来读取每个图片对应的标签
  59. # 请查看解压后的验证集标签数据,观察csvfile文件里面所包含的内容
  60. # csvfile文件所包含的内容格式如下,每一行代表一个样本,
  61. # 其中第一列是图片id,第二列是文件名,第三列是图片标签,
  62. # 第四列和第五列是Fovea的坐标,与分类任务无关
  63. # ID,imgName,Label,Fovea_X,Fovea_Y
  64. # 1,V0001.jpg,0,1157.74,1019.87
  65. # 2,V0002.jpg,1,1285.82,1080.47
  66. # 打开包含验证集标签的csvfile,并读入其中的内容
  67. filelists = open(csvfile).readlines()
  68. def reader():
  69. batch_imgs = []
  70. batch_labels = []
  71. for line in filelists[1:]:
  72. line = line.strip().split(',')
  73. name = line[1]
  74. label = int(line[2])
  75. # 根据图片文件名加载图片,并对图像数据作预处理
  76. filepath = os.path.join(datadir, name)
  77. img = cv2.imread(filepath)
  78. img = transform_img(img)
  79. # 每读取一个样本的数据,就将其放入数据列表中
  80. batch_imgs.append(img)
  81. batch_labels.append(label)
  82. if len(batch_imgs) == batch_size:
  83. # 当数据列表的长度等于batch_size的时候,
  84. # 把这些数据当作一个mini-batch,并作为数据生成器的一个输出
  85. imgs_array = np.array(batch_imgs).astype('float32')
  86. labels_array = np.array(batch_labels).astype('float32').reshape(-1, 1)
  87. yield imgs_array, labels_array
  88. batch_imgs = []
  89. batch_labels = []
  90. if len(batch_imgs) > 0:
  91. # 剩余样本数目不足一个batch_size的数据,一起打包成一个mini-batch
  92. imgs_array = np.array(batch_imgs).astype('float32')
  93. labels_array = np.array(batch_labels).astype('float32').reshape(-1, 1)
  94. yield imgs_array, labels_array
  95. return reader
  1. # 查看数据形状
  2. DATADIR = '/home/aistudio/work/palm/PALM-Training400/PALM-Training400'
  3. train_loader = data_loader(DATADIR,
  4. batch_size=10, mode='train')
  5. data_reader = train_loader()
  6. data = next(data_reader)
  7. data[0].shape, data[1].shape
  1. ((10, 3, 224, 224), (10, 1))

启动训练

  1. # -*- coding: utf-8 -*-
  2. # LeNet 识别眼疾图片
  3. import os
  4. import random
  5. import paddle
  6. import paddle.fluid as fluid
  7. import numpy as np
  8. DATADIR = '/home/aistudio/work/palm/PALM-Training400/PALM-Training400'
  9. DATADIR2 = '/home/aistudio/work/palm/PALM-Validation400'
  10. CSVFILE = '/home/aistudio/work/palm/PALM-Validation-GT/labels.csv'
  11. # 定义训练过程
  12. def train(model):
  13. with fluid.dygraph.guard():
  14. print('start training ... ')
  15. model.train()
  16. epoch_num = 5
  17. # 定义优化器
  18. opt = fluid.optimizer.Momentum(learning_rate=0.001, momentum=0.9, parameter_list=model.parameters())
  19. # 定义数据读取器,训练数据读取器和验证数据读取器
  20. train_loader = data_loader(DATADIR, batch_size=10, mode='train')
  21. valid_loader = valid_data_loader(DATADIR2, CSVFILE)
  22. for epoch in range(epoch_num):
  23. for batch_id, data in enumerate(train_loader()):
  24. x_data, y_data = data
  25. img = fluid.dygraph.to_variable(x_data)
  26. label = fluid.dygraph.to_variable(y_data)
  27. # 运行模型前向计算,得到预测值
  28. logits = model(img)
  29. # 进行loss计算
  30. loss = fluid.layers.sigmoid_cross_entropy_with_logits(logits, label)
  31. avg_loss = fluid.layers.mean(loss)
  32. if batch_id % 10 == 0:
  33. print("epoch: {}, batch_id: {}, loss is: {}".format(epoch, batch_id, avg_loss.numpy()))
  34. # 反向传播,更新权重,清除梯度
  35. avg_loss.backward()
  36. opt.minimize(avg_loss)
  37. model.clear_gradients()
  38. model.eval()
  39. accuracies = []
  40. losses = []
  41. for batch_id, data in enumerate(valid_loader()):
  42. x_data, y_data = data
  43. img = fluid.dygraph.to_variable(x_data)
  44. label = fluid.dygraph.to_variable(y_data)
  45. # 运行模型前向计算,得到预测值
  46. logits = model(img)
  47. # 二分类,sigmoid计算后的结果以0.5为阈值分两个类别
  48. # 计算sigmoid后的预测概率,进行loss计算
  49. pred = fluid.layers.sigmoid(logits)
  50. loss = fluid.layers.sigmoid_cross_entropy_with_logits(logits, label)
  51. # 计算预测概率小于0.5的类别
  52. pred2 = pred * (-1.0) + 1.0
  53. # 得到两个类别的预测概率,并沿第一个维度级联
  54. pred = fluid.layers.concat([pred2, pred], axis=1)
  55. acc = fluid.layers.accuracy(pred, fluid.layers.cast(label, dtype='int64'))
  56. accuracies.append(acc.numpy())
  57. losses.append(loss.numpy())
  58. print("[validation] accuracy/loss: {}/{}".format(np.mean(accuracies), np.mean(losses)))
  59. model.train()
  60. # save params of model
  61. fluid.save_dygraph(model.state_dict(), 'mnist')
  62. # save optimizer state
  63. fluid.save_dygraph(opt.state_dict(), 'mnist')
  64. # 定义评估过程
  65. def evaluation(model, params_file_path):
  66. with fluid.dygraph.guard():
  67. print('start evaluation .......')
  68. #加载模型参数
  69. model_state_dict, _ = fluid.load_dygraph(params_file_path)
  70. model.load_dict(model_state_dict)
  71. model.eval()
  72. eval_loader = load_data('eval')
  73. acc_set = []
  74. avg_loss_set = []
  75. for batch_id, data in enumerate(eval_loader()):
  76. x_data, y_data = data
  77. img = fluid.dygraph.to_variable(x_data)
  78. label = fluid.dygraph.to_variable(y_data)
  79. # 计算预测和精度
  80. prediction, acc = model(img, label)
  81. # 计算损失函数值
  82. loss = fluid.layers.cross_entropy(input=prediction, label=label)
  83. avg_loss = fluid.layers.mean(loss)
  84. acc_set.append(float(acc.numpy()))
  85. avg_loss_set.append(float(avg_loss.numpy()))
  86. # 求平均精度
  87. acc_val_mean = np.array(acc_set).mean()
  88. avg_loss_val_mean = np.array(avg_loss_set).mean()
  89. print('loss={}, acc={}'.format(avg_loss_val_mean, acc_val_mean))
  90. # 导入需要的包
  91. import paddle
  92. import paddle.fluid as fluid
  93. import numpy as np
  94. from paddle.fluid.dygraph.nn import Conv2D, Pool2D, Linear
  95. # 定义 LeNet 网络结构
  96. class LeNet(fluid.dygraph.Layer):
  97. def __init__(self, name_scope, num_classes=1):
  98. super(LeNet, self).__init__(name_scope)
  99. # 创建卷积和池化层块,每个卷积层使用Sigmoid激活函数,后面跟着一个2x2的池化
  100. self.conv1 = Conv2D(num_channels=3, num_filters=6, filter_size=5, act='sigmoid')
  101. self.pool1 = Pool2D(pool_size=2, pool_stride=2, pool_type='max')
  102. self.conv2 = Conv2D(num_channels=6, num_filters=16, filter_size=5, act='sigmoid')
  103. self.pool2 = Pool2D(pool_size=2, pool_stride=2, pool_type='max')
  104. # 创建第3个卷积层
  105. self.conv3 = Conv2D(num_channels=16, num_filters=120, filter_size=4, act='sigmoid')
  106. # 创建全连接层,第一个全连接层的输出神经元个数为64, 第二个全连接层输出神经元个数为分裂标签的类别数
  107. self.fc1 = Linear(input_dim=300000, output_dim=64, act='sigmoid')
  108. self.fc2 = Linear(input_dim=64, output_dim=num_classes)
  109. # 网络的前向计算过程
  110. def forward(self, x):
  111. x = self.conv1(x)
  112. x = self.pool1(x)
  113. x = self.conv2(x)
  114. x = self.pool2(x)
  115. x = self.conv3(x)
  116. x = fluid.layers.reshape(x, [x.shape[0], -1])
  117. x = self.fc1(x)
  118. x = self.fc2(x)
  119. return x
  120. if __name__ == '__main__':
  121. # 创建模型
  122. with fluid.dygraph.guard():
  123. model = LeNet("LeNet_", num_classes=1)
  124. train(model)

通过运行结果可以看出,在眼疾筛查数据集iChallenge-PM上,LeNet的loss很难下降,模型没有收敛。这是因为MNIST数据集的图片尺寸比较小(

图像分类 - 图9 ),但是眼疾筛查数据集图片尺寸比较大(原始图片尺寸约为 图像分类 - 图10 ,经过缩放之后变成 图像分类 - 图11 ),LeNet模型很难进行有效分类。这说明在图片尺寸比较大时,LeNet在图像分类任务上存在局限性。

AlexNet

通过上面的实际训练可以看到,虽然LeNet在手写数字识别数据集上取得了很好的结果,但在更大的数据集上表现却并不好。自从1998年LeNet问世以来,接下来十几年的时间里,神经网络并没有在计算机视觉领域取得很好的结果,反而一度被其它算法所超越,原因主要有两方面,一是神经网络的计算比较复杂,对当时计算机的算力来说,训练神经网络是件非常耗时的事情;另一方面,当时还没有专门针对神经网络做算法和训练技巧的优化,神经网络的收敛性是件非常困难的事情。

随着技术的进步和发展,计算机的算力越来越强大,尤其是在GPU并行计算能力的推动下,复杂神经网络的计算也变得更加容易实施。另一方面,互联网上涌现出越来越多的数据,极大的丰富了数据库。同时也有越来越多的研究人员开始专门针对神经网络做算法和模型的优化,Alex Krizhevsky等人提出的AlexNet以很大优势获得了2012年ImageNet比赛的冠军。这一成果极大的激发了业界对神经网络的兴趣,开创了使用深度神经网络解决图像问题的途径,随后也在这一领域涌现出越来越多的优秀工作。

AlexNet与LeNet相比,具有更深的网络结构,包含5层卷积和3层全连接,同时使用了如下三种方法改进模型的训练过程:

  • 数据增多:深度学习中常用的一种处理方式,通过对训练随机加一些变化,比如平移、缩放、裁剪、旋转、翻转或者增减亮度等,产生一系列跟原始图片相似但又不完全相同的样本,从而扩大训练数据集。通过这种方式,可以随机改变训练样本,避免模型过度依赖于某些属性,能从一定程度上抑制过拟合。

  • 使用Dropout抑制过拟合

  • 使用ReLU激活函数少梯度消失现象


说明:

下一节详细介绍数据增多的具体实现方式。


AlexNet的具体结构如 图2 所示:

图像分类 - 图12

图2:AlexNet模型网络结构示意图

AlexNet在眼疾筛查数据集iChallenge-PM上具体实现的代码如下所示:

  1. # -*- coding:utf-8 -*-
  2. # 导入需要的包
  3. import paddle
  4. import paddle.fluid as fluid
  5. import numpy as np
  6. from paddle.fluid.dygraph.nn import Conv2D, Pool2D, Linear
  7. # 定义 AlexNet 网络结构
  8. class AlexNet(fluid.dygraph.Layer):
  9. def __init__(self, name_scope, num_classes=1):
  10. super(AlexNet, self).__init__(name_scope)
  11. name_scope = self.full_name()
  12. # AlexNet与LeNet一样也会同时使用卷积和池化层提取图像特征
  13. # 与LeNet不同的是激活函数换成了‘relu’
  14. self.conv1 = Conv2D(num_channels=3, num_filters=96, filter_size=11, stride=4, padding=5, act='relu')
  15. self.pool1 = Pool2D(pool_size=2, pool_stride=2, pool_type='max')
  16. self.conv2 = Conv2D(num_channels=96, num_filters=256, filter_size=5, stride=1, padding=2, act='relu')
  17. self.pool2 = Pool2D(pool_size=2, pool_stride=2, pool_type='max')
  18. self.conv3 = Conv2D(num_channels=256, num_filters=384, filter_size=3, stride=1, padding=1, act='relu')
  19. self.conv4 = Conv2D(num_channels=384, num_filters=384, filter_size=3, stride=1, padding=1, act='relu')
  20. self.conv5 = Conv2D(num_channels=384, num_filters=256, filter_size=3, stride=1, padding=1, act='relu')
  21. self.pool5 = Pool2D(pool_size=2, pool_stride=2, pool_type='max')
  22. self.fc1 = Linear(input_dim=12544, output_dim=4096, act='relu')
  23. self.drop_ratio1 = 0.5
  24. self.fc2 = Linear(input_dim=4096, output_dim=4096, act='relu')
  25. self.drop_ratio2 = 0.5
  26. self.fc3 = Linear(input_dim=4096, output_dim=num_classes)
  27. def forward(self, x):
  28. x = self.conv1(x)
  29. x = self.pool1(x)
  30. x = self.conv2(x)
  31. x = self.pool2(x)
  32. x = self.conv3(x)
  33. x = self.conv4(x)
  34. x = self.conv5(x)
  35. x = self.pool5(x)
  36. x = fluid.layers.reshape(x, [x.shape[0], -1])
  37. x = self.fc1(x)
  38. # 在全连接之后使用dropout抑制过拟合
  39. x= fluid.layers.dropout(x, self.drop_ratio1)
  40. x = self.fc2(x)
  41. # 在全连接之后使用dropout抑制过拟合
  42. x = fluid.layers.dropout(x, self.drop_ratio2)
  43. x = self.fc3(x)
  44. return x
  1. with fluid.dygraph.guard():
  2. model = AlexNet("AlexNet")
  3. train(model)

通过运行结果可以发现,在眼疾筛查数据集iChallenge-PM上使用AlexNet,loss能有效下降,经过5个epoch的训练,在验证集上的准确率可以达到94%左右。

VGG

VGG是当前最流行的CNN模型之一,2014年由Simonyan和Zisserman提出,其命名来源于论文作者所在的实验室Visual Geometry Group。AlexNet模型通过构造多层网络,取得了较好的效果,但是并没有给出深度神经网络设计的方向。VGG通过使用一系列大小为3x3的小尺寸卷积核和pooling层构造深度卷积神经网络,并取得了较好的效果。VGG模型因为结构简单、应用性极强而广受研究者欢迎,尤其是它的网络结构设计方法,为构建深度神经网络提供了方向。

图3 是VGG-16的网络结构示意图,有13层卷积和3层全连接层。VGG网络的设计严格使用

图像分类 - 图13 的卷积层和池化层来提取特征,并在网络的最后面使用三层全连接层,将最后一层全连接层的输出作为分类的预测。 在VGG中每层卷积将使用ReLU作为激活函数,在全连接层之后添加dropout来抑制过拟合。使用小的卷积核能够有效地减少参数的个数,使得训练和测试变得更加有效。比如使用两层 图像分类 - 图14 卷积层,可以得到感受野为5的特征图,而比使用 图像分类 - 图15 的卷积层需要更少的参数。由于卷积核比较小,可以堆叠更多的卷积层,加深网络的深度,这对于图像分类任务来说是有利的。VGG模型的成功证明了增加网络的深度,可以更好的学习图像中的特征模式。

图像分类 - 图16

图3:VGG模型网络结构示意图

VGG在眼疾识别数据集iChallenge-PM上的具体实现如下代码所示:

  1. # -*- coding:utf-8 -*-
  2. # VGG模型代码
  3. import numpy as np
  4. import paddle
  5. import paddle.fluid as fluid
  6. from paddle.fluid.layer_helper import LayerHelper
  7. from paddle.fluid.dygraph.nn import Conv2D, Pool2D, BatchNorm, Linear
  8. from paddle.fluid.dygraph.base import to_variable
  9. # 定义vgg块,包含多层卷积和1层2x2的最大池化层
  10. class vgg_block(fluid.dygraph.Layer):
  11. def __init__(self, name_scope, num_convs, in_channels, out_channels):
  12. """
  13. num_convs, 卷积层的数目
  14. num_channels, 卷积层的输出通道数,在同一个Incepition块内,卷积层输出通道数是一样的
  15. """
  16. super(vgg_block, self).__init__(name_scope)
  17. self.conv_list = []
  18. for i in range(num_convs):
  19. conv_layer = self.add_sublayer('conv_' + str(i), Conv2D(num_channels=in_channels,
  20. num_filters=out_channels, filter_size=3, padding=1, act='relu'))
  21. self.conv_list.append(conv_layer)
  22. in_channels = out_channels
  23. self.pool = Pool2D(pool_stride=2, pool_size = 2, pool_type='max')
  24. def forward(self, x):
  25. for item in self.conv_list:
  26. x = item(x)
  27. return self.pool(x)
  28. class VGG(fluid.dygraph.Layer):
  29. def __init__(self, name_scope, conv_arch=((2, 64),
  30. (2, 128), (3, 256), (3, 512), (3, 512))):
  31. super(VGG, self).__init__(name_scope)
  32. self.vgg_blocks=[]
  33. iter_id = 0
  34. # 添加vgg_block
  35. # 这里一共5个vgg_block,每个block里面的卷积层数目和输出通道数由conv_arch指定
  36. in_channels = [3, 64, 128, 256, 512, 512]
  37. for (num_convs, num_channels) in conv_arch:
  38. block = self.add_sublayer('block_' + str(iter_id),
  39. vgg_block(self.full_name(), num_convs, in_channels=in_channels[iter_id],
  40. out_channels=num_channels))
  41. self.vgg_blocks.append(block)
  42. iter_id += 1
  43. self.fc1 = Linear(input_dim=512*7*7, output_dim=4096,
  44. act='relu')
  45. self.drop1_ratio = 0.5
  46. self.fc2= Linear(input_dim=4096, output_dim=4096,
  47. act='relu')
  48. self.drop2_ratio = 0.5
  49. self.fc3 = Linear(input_dim=4096, output_dim=1)
  50. def forward(self, x):
  51. for item in self.vgg_blocks:
  52. x = item(x)
  53. x = fluid.layers.reshape(x, [x.shape[0], -1])
  54. x = fluid.layers.dropout(self.fc1(x), self.drop1_ratio)
  55. x = fluid.layers.dropout(self.fc2(x), self.drop2_ratio)
  56. x = self.fc3(x)
  57. return x
  1. with fluid.dygraph.guard():
  2. model = VGG("VGG")
  3. train(model)

通过运行结果可以发现,在眼疾筛查数据集iChallenge-PM上使用VGG,loss能有效的下降,经过5个epoch的训练,在验证集上的准确率可以达到94%左右。

GoogLeNet

GoogLeNet是2014年ImageNet比赛的冠军,它的主要特点是网络不仅有深度,还在横向上具有“宽度”。由于图像信息在空间尺寸上的巨大差异,如何选择合适的卷积核大小来提取特征就显得比较困难了。空间分布范围更广的图像信息适合用较大的卷积核来提取其特征,而空间分布范围较小的图像信息则适合用较小的卷积核来提取其特征。为了解决这个问题,GoogLeNet提出了一种被称为Inception模块的方案。如 图4 所示:


说明:

  • Google的研究人员为了向LeNet致敬,特地将模型命名为GoogLeNet
  • Inception一词来源于电影《盗梦空间》(Inception)

图像分类 - 图17

图4:Inception模块结构示意图

图4(a)是Inception模块的设计思想,使用3个不同大小的卷积核对输入图片进行卷积操作,并附加最大池化,将这4个操作的输出沿着通道这一维度进行拼接,构成的输出特征图将会包含经过不同大小的卷积核提取出来的特征。Inception模块采用多通路(multi-path)的设计形式,每个支路使用不同大小的卷积核,最终输出特征图的通道数是每个支路输出通道数的总和,这将会导致输出通道数变得很大,尤其是使用多个Inception模块串联操作的时候,模型参数量会变得非常巨大。为了减小参数量,Inception模块使用了图(b)中的设计方式,在每个3x3和5x5的卷积层之前,增加1x1的卷积层来控制输出通道数;在最大池化层后面增加1x1卷积层减小输出通道数。基于这一设计思想,形成了上图(b)中所示的结构。下面这段程序是Inception块的具体实现方式,可以对照图(b)和代码一起阅读。


提示:

可能有读者会问,经过3x3的最大池化之后图像尺寸不会减小吗,为什么还能跟另外3个卷积输出的特征图进行拼接?这是因为池化操作可以指定窗口大小

图像分类 - 图18 ,pool_stride=1和pool_padding=1,输出特征图尺寸可以保持不变。


Inception模块的具体实现如下代码所示:

  1. class Inception(fluid.dygraph.Layer):
  2. def __init__(self, name_scope, c1, c2, c3, c4, **kwargs):
  3. '''
  4. Inception模块的实现代码,
  5. name_scope, 模块名称,数据类型为string
  6. c1, 图(b)中第一条支路1x1卷积的输出通道数,数据类型是整数
  7. c2,图(b)中第二条支路卷积的输出通道数,数据类型是tuple或list,
  8. 其中c2[0]是1x1卷积的输出通道数,c2[1]是3x3
  9. c3,图(b)中第三条支路卷积的输出通道数,数据类型是tuple或list,
  10. 其中c3[0]是1x1卷积的输出通道数,c3[1]是3x3
  11. c4, 图(b)中第一条支路1x1卷积的输出通道数,数据类型是整数
  12. '''
  13. super(Inception, self).__init__(name_scope)
  14. # 依次创建Inception块每条支路上使用到的操作
  15. self.p1_1 = Conv2D(self.full_name(), num_filters=c1,
  16. filter_size=1, act='relu')
  17. self.p2_1 = Conv2D(self.full_name(), num_filters=c2[0],
  18. filter_size=1, act='relu')
  19. self.p2_2 = Conv2D(self.full_name(), num_filters=c2[1],
  20. filter_size=3, padding=1, act='relu')
  21. self.p3_1 = Conv2D(self.full_name(), num_filters=c3[0],
  22. filter_size=1, act='relu')
  23. self.p3_2 = Conv2D(self.full_name(), num_filters=c3[1],
  24. filter_size=5, padding=2, act='relu')
  25. self.p4_1 = Pool2D(self.full_name(), pool_size=3,
  26. pool_stride=1, pool_padding=1,
  27. pool_type='max')
  28. self.p4_2 = Conv2D(self.full_name(), num_filters=c4,
  29. filter_size=1, act='relu')
  30. def forward(self, x):
  31. # 支路1只包含一个1x1卷积
  32. p1 = self.p1_1(x)
  33. # 支路2包含 1x1卷积 + 3x3卷积
  34. p2 = self.p2_2(self.p2_1(x))
  35. # 支路3包含 1x1卷积 + 5x5卷积
  36. p3 = self.p3_2(self.p3_1(x))
  37. # 支路4包含 最大池化和1x1卷积
  38. p4 = self.p4_2(self.p4_1(x))
  39. # 将每个支路的输出特征图拼接在一起作为最终的输出结果
  40. return fluid.layers.concat([p1, p2, p3, p4], axis=1)

GoogLeNet的架构如 图5 所示,在主体卷积部分中使用5个模块(block),每个模块之间使用步幅为2的3 ×3最大池化层来减小输出高宽。

  • 第一模块使用一个64通道的7 × 7卷积层。
  • 第二模块使用2个卷积层:首先是64通道的1 × 1卷积层,然后是将通道增大3倍的3 × 3卷积层。
  • 第三模块串联2个完整的Inception块。
  • 第四模块串联了5个Inception块。
  • 第五模块串联了2 个Inception块。
  • 第五模块的后面紧跟输出层,使用全局平均池化 层来将每个通道的高和宽变成1,最后接上一个输出个数为标签类别数的全连接层。

说明: 在原作者的论文中添加了图中所示的softmax1和softmax2两个辅助分类器,如下图所示,训练时将三个分类器的损失函数进行加权求和,以缓解梯度消失现象。这里的程序作了简化,没有加入辅助分类器。


图像分类 - 图19

图5:GoogLeNet模型网络结构示意图

GoogLeNet的具体实现如下代码所示:

  1. # -*- coding:utf-8 -*-
  2. # GoogLeNet模型代码
  3. import numpy as np
  4. import paddle
  5. import paddle.fluid as fluid
  6. from paddle.fluid.layer_helper import LayerHelper
  7. from paddle.fluid.dygraph.nn import Conv2D, Pool2D, BatchNorm, Linear
  8. from paddle.fluid.dygraph.base import to_variable
  9. # 定义Inception块
  10. class Inception(fluid.dygraph.Layer):
  11. def __init__(self, name_scope, c0,c1, c2, c3, c4, **kwargs):
  12. '''
  13. Inception模块的实现代码,
  14. name_scope, 模块名称,数据类型为string
  15. c1, 图(b)中第一条支路1x1卷积的输出通道数,数据类型是整数
  16. c2,图(b)中第二条支路卷积的输出通道数,数据类型是tuple或list,
  17. 其中c2[0]是1x1卷积的输出通道数,c2[1]是3x3
  18. c3,图(b)中第三条支路卷积的输出通道数,数据类型是tuple或list,
  19. 其中c3[0]是1x1卷积的输出通道数,c3[1]是3x3
  20. c4, 图(b)中第一条支路1x1卷积的输出通道数,数据类型是整数
  21. '''
  22. super(Inception, self).__init__(name_scope)
  23. # 依次创建Inception块每条支路上使用到的操作
  24. self.p1_1 = Conv2D(num_channels=c0, num_filters=c1,
  25. filter_size=1, act='relu')
  26. self.p2_1 = Conv2D(num_channels=c0, num_filters=c2[0],
  27. filter_size=1, act='relu')
  28. self.p2_2 = Conv2D(num_channels=c2[0], num_filters=c2[1],
  29. filter_size=3, padding=1, act='relu')
  30. self.p3_1 = Conv2D(num_channels=c0, num_filters=c3[0],
  31. filter_size=1, act='relu')
  32. self.p3_2 = Conv2D(num_channels=c3[0], num_filters=c3[1],
  33. filter_size=5, padding=2, act='relu')
  34. self.p4_1 = Pool2D(pool_size=3,
  35. pool_stride=1, pool_padding=1,
  36. pool_type='max')
  37. self.p4_2 = Conv2D(num_channels=c0, num_filters=c4,
  38. filter_size=1, act='relu')
  39. def forward(self, x):
  40. # 支路1只包含一个1x1卷积
  41. p1 = self.p1_1(x)
  42. # 支路2包含 1x1卷积 + 3x3卷积
  43. p2 = self.p2_2(self.p2_1(x))
  44. # 支路3包含 1x1卷积 + 5x5卷积
  45. p3 = self.p3_2(self.p3_1(x))
  46. # 支路4包含 最大池化和1x1卷积
  47. p4 = self.p4_2(self.p4_1(x))
  48. # 将每个支路的输出特征图拼接在一起作为最终的输出结果
  49. return fluid.layers.concat([p1, p2, p3, p4], axis=1)
  50. class GoogLeNet(fluid.dygraph.Layer):
  51. def __init__(self, name_scope):
  52. super(GoogLeNet, self).__init__(name_scope)
  53. # GoogLeNet包含五个模块,每个模块后面紧跟一个池化层
  54. # 第一个模块包含1个卷积层
  55. self.conv1 = Conv2D(num_channels=3, num_filters=64, filter_size=7,
  56. padding=3, act='relu')
  57. # 3x3最大池化
  58. self.pool1 = Pool2D(pool_size=3, pool_stride=2,
  59. pool_padding=1, pool_type='max')
  60. # 第二个模块包含2个卷积层
  61. self.conv2_1 = Conv2D(num_channels=64, num_filters=64,
  62. filter_size=1, act='relu')
  63. self.conv2_2 = Conv2D(num_channels=64, num_filters=192,
  64. filter_size=3, padding=1, act='relu')
  65. # 3x3最大池化
  66. self.pool2 = Pool2D(pool_size=3, pool_stride=2,
  67. pool_padding=1, pool_type='max')
  68. # 第三个模块包含2个Inception块
  69. self.block3_1 = Inception(self.full_name(), 192, 64, (96, 128), (16, 32), 32)
  70. self.block3_2 = Inception(self.full_name(), 256, 128, (128, 192), (32, 96), 64)
  71. # 3x3最大池化
  72. self.pool3 = Pool2D(pool_size=3, pool_stride=2,
  73. pool_padding=1, pool_type='max')
  74. # 第四个模块包含5个Inception块
  75. self.block4_1 = Inception(self.full_name(), 480, 192, (96, 208), (16, 48), 64)
  76. self.block4_2 = Inception(self.full_name(), 512, 160, (112, 224), (24, 64), 64)
  77. self.block4_3 = Inception(self.full_name(), 512, 128, (128, 256), (24, 64), 64)
  78. self.block4_4 = Inception(self.full_name(), 512, 112, (144, 288), (32, 64), 64)
  79. self.block4_5 = Inception(self.full_name(), 528, 256, (160, 320), (32, 128), 128)
  80. # 3x3最大池化
  81. self.pool4 = Pool2D(pool_size=3, pool_stride=2,
  82. pool_padding=1, pool_type='max')
  83. # 第五个模块包含2个Inception块
  84. self.block5_1 = Inception(self.full_name(), 832, 256, (160, 320), (32, 128), 128)
  85. self.block5_2 = Inception(self.full_name(), 832, 384, (192, 384), (48, 128), 128)
  86. # 全局池化,尺寸用的是global_pooling,pool_stride不起作用
  87. self.pool5 = Pool2D(pool_stride=1,
  88. global_pooling=True, pool_type='avg')
  89. self.fc = Linear(input_dim=1024, output_dim=1, act=None)
  90. def forward(self, x):
  91. x = self.pool1(self.conv1(x))
  92. x = self.pool2(self.conv2_2(self.conv2_1(x)))
  93. x = self.pool3(self.block3_2(self.block3_1(x)))
  94. x = self.block4_3(self.block4_2(self.block4_1(x)))
  95. x = self.pool4(self.block4_5(self.block4_4(x)))
  96. x = self.pool5(self.block5_2(self.block5_1(x)))
  97. x = fluid.layers.reshape(x, [x.shape[0], -1])
  98. x = self.fc(x)
  99. return x
  1. with fluid.dygraph.guard():
  2. model = GoogLeNet("GoogLeNet")
  3. train(model)

通过运行结果可以发现,使用GoogLeNet在眼疾筛查数据集iChallenge-PM上,loss能有效的下降,经过5个epoch的训练,在验证集上的准确率可以达到95%左右。

ResNet

ResNet是2015年ImageNet比赛的冠军,将识别错误率降低到了3.6%,这个结果甚至超出了正常人眼识别的精度。

通过前面几个经典模型学习,我们可以发现随着深度学习的不断发展,模型的层数越来越多,网络结构也越来越复杂。那么是否加深网络结构,就一定会得到更好的效果呢?从理论上来说,假设新增加的层都是恒等映射,只要原有的层学出跟原模型一样的参数,那么深模型结构就能达到原模型结构的效果。换句话说,原模型的解只是新模型的解的子空间,在新模型解的空间里应该能找到比原模型解对应的子空间更好的结果。但是实践表明,增加网络的层数之后,训练误差往往不降反升。

Kaiming He等人提出了残差网络ResNet来解决上述问题,其基本思想如 图6所示。

  • 图6(a):表示增加网络的时候,将x映射成 图像分类 - 图20 输出。
  • 图6(b):对图6(a)作了改进,输出 图像分类 - 图21 。这时不是直接学习输出特征y的表示,而是学习 图像分类 - 图22
    • 如果想学习出原模型的表示,只需将F(x)的参数全部设置为0,则 图像分类 - 图23 是恒等映射。
    • 图像分类 - 图24 也叫做残差项,如果 图像分类 - 图25 的映射接近恒等映射,图6(b)中通过学习残差项也比图6(a)学习完整映射形式更加容易。

图像分类 - 图26

图6:残差块设计思想

图6(b)的结构是残差网络的基础,这种结构也叫做残差块(residual block)。输入x通过跨层连接,能更快的向前传播数据,或者向后传播梯度。残差块的具体设计方案如 7 所示,这种设计方案也成称作瓶颈结构(BottleNeck)。

图像分类 - 图27

图7:残差块结构示意图

下图表示出了ResNet-50的结构,一共包含49层卷积和1层全连接,所以被称为ResNet-50。

图像分类 - 图28

图8:ResNet-50模型网络结构示意图

ResNet-50的具体实现如下代码所示:

  1. # -*- coding:utf-8 -*-
  2. # ResNet模型代码
  3. import numpy as np
  4. import paddle
  5. import paddle.fluid as fluid
  6. from paddle.fluid.layer_helper import LayerHelper
  7. from paddle.fluid.dygraph.nn import Conv2D, Pool2D, BatchNorm, Linear
  8. from paddle.fluid.dygraph.base import to_variable
  9. # ResNet中使用了BatchNorm层,在卷积层的后面加上BatchNorm以提升数值稳定性
  10. # 定义卷积批归一化块
  11. class ConvBNLayer(fluid.dygraph.Layer):
  12. def __init__(self,
  13. num_channels,
  14. num_filters,
  15. filter_size,
  16. stride=1,
  17. groups=1,
  18. act=None):
  19. """
  20. name_scope, 模块的名字
  21. num_channels, 卷积层的输入通道数
  22. num_filters, 卷积层的输出通道数
  23. stride, 卷积层的步幅
  24. groups, 分组卷积的组数,默认groups=1不使用分组卷积
  25. act, 激活函数类型,默认act=None不使用激活函数
  26. """
  27. super(ConvBNLayer, self).__init__()
  28. # 创建卷积层
  29. self._conv = Conv2D(
  30. num_channels=num_channels,
  31. num_filters=num_filters,
  32. filter_size=filter_size,
  33. stride=stride,
  34. padding=(filter_size - 1) // 2,
  35. groups=groups,
  36. act=None,
  37. bias_attr=False)
  38. # 创建BatchNorm层
  39. self._batch_norm = BatchNorm(num_filters, act=act)
  40. def forward(self, inputs):
  41. y = self._conv(inputs)
  42. y = self._batch_norm(y)
  43. return y
  44. # 定义残差块
  45. # 每个残差块会对输入图片做三次卷积,然后跟输入图片进行短接
  46. # 如果残差块中第三次卷积输出特征图的形状与输入不一致,则对输入图片做1x1卷积,将其输出形状调整成一致
  47. class BottleneckBlock(fluid.dygraph.Layer):
  48. def __init__(self,
  49. name_scope,
  50. num_channels,
  51. num_filters,
  52. stride,
  53. shortcut=True):
  54. super(BottleneckBlock, self).__init__(name_scope)
  55. # 创建第一个卷积层 1x1
  56. self.conv0 = ConvBNLayer(
  57. num_channels=num_channels,
  58. num_filters=num_filters,
  59. filter_size=1,
  60. act='relu')
  61. # 创建第二个卷积层 3x3
  62. self.conv1 = ConvBNLayer(
  63. num_channels=num_filters,
  64. num_filters=num_filters,
  65. filter_size=3,
  66. stride=stride,
  67. act='relu')
  68. # 创建第三个卷积 1x1,但输出通道数乘以4
  69. self.conv2 = ConvBNLayer(
  70. num_channels=num_filters,
  71. num_filters=num_filters * 4,
  72. filter_size=1,
  73. act=None)
  74. # 如果conv2的输出跟此残差块的输入数据形状一致,则shortcut=True
  75. # 否则shortcut = False,添加1个1x1的卷积作用在输入数据上,使其形状变成跟conv2一致
  76. if not shortcut:
  77. self.short = ConvBNLayer(
  78. num_channels=num_channels,
  79. num_filters=num_filters * 4,
  80. filter_size=1,
  81. stride=stride)
  82. self.shortcut = shortcut
  83. self._num_channels_out = num_filters * 4
  84. def forward(self, inputs):
  85. y = self.conv0(inputs)
  86. conv1 = self.conv1(y)
  87. conv2 = self.conv2(conv1)
  88. # 如果shortcut=True,直接将inputs跟conv2的输出相加
  89. # 否则需要对inputs进行一次卷积,将形状调整成跟conv2输出一致
  90. if self.shortcut:
  91. short = inputs
  92. else:
  93. short = self.short(inputs)
  94. y = fluid.layers.elementwise_add(x=short, y=conv2)
  95. layer_helper = LayerHelper(self.full_name(), act='relu')
  96. return layer_helper.append_activation(y)
  97. # 定义ResNet模型
  98. class ResNet(fluid.dygraph.Layer):
  99. def __init__(self, name_scope, layers=50, class_dim=1):
  100. """
  101. name_scope,模块名称
  102. layers, 网络层数,可以是50, 101或者152
  103. class_dim,分类标签的类别数
  104. """
  105. super(ResNet, self).__init__(name_scope)
  106. self.layers = layers
  107. supported_layers = [50, 101, 152]
  108. assert layers in supported_layers, \
  109. "supported layers are {} but input layer is {}".format(supported_layers, layers)
  110. if layers == 50:
  111. #ResNet50包含多个模块,其中第2到第5个模块分别包含3、4、6、3个残差块
  112. depth = [3, 4, 6, 3]
  113. elif layers == 101:
  114. #ResNet101包含多个模块,其中第2到第5个模块分别包含3、4、23、3个残差块
  115. depth = [3, 4, 23, 3]
  116. elif layers == 152:
  117. #ResNet50包含多个模块,其中第2到第5个模块分别包含3、8、36、3个残差块
  118. depth = [3, 8, 36, 3]
  119. # 残差块中使用到的卷积的输出通道数
  120. num_filters = [64, 128, 256, 512]
  121. # ResNet的第一个模块,包含1个7x7卷积,后面跟着1个最大池化层
  122. self.conv = ConvBNLayer(
  123. num_channels=3,
  124. num_filters=64,
  125. filter_size=7,
  126. stride=2,
  127. act='relu')
  128. self.pool2d_max = Pool2D(
  129. pool_size=3,
  130. pool_stride=2,
  131. pool_padding=1,
  132. pool_type='max')
  133. # ResNet的第二到第五个模块c2、c3、c4、c5
  134. self.bottleneck_block_list = []
  135. num_channels = 64
  136. for block in range(len(depth)):
  137. shortcut = False
  138. for i in range(depth[block]):
  139. bottleneck_block = self.add_sublayer(
  140. 'bb_%d_%d' % (block, i),
  141. BottleneckBlock(
  142. self.full_name(),
  143. num_channels=num_channels,
  144. num_filters=num_filters[block],
  145. stride=2 if i == 0 and block != 0 else 1, # c3、c4、c5将会在第一个残差块使用stride=2;其余所有残差块stride=1
  146. shortcut=shortcut))
  147. num_channels = bottleneck_block._num_channels_out
  148. self.bottleneck_block_list.append(bottleneck_block)
  149. shortcut = True
  150. # 在c5的输出特征图上使用全局池化
  151. self.pool2d_avg = Pool2D(pool_size=7, pool_type='avg', global_pooling=True)
  152. # stdv用来作为全连接层随机初始化参数的方差
  153. import math
  154. stdv = 1.0 / math.sqrt(2048 * 1.0)
  155. # 创建全连接层,输出大小为类别数目
  156. self.out = Linear(input_dim=2048, output_dim=class_dim,
  157. param_attr=fluid.param_attr.ParamAttr(
  158. initializer=fluid.initializer.Uniform(-stdv, stdv)))
  159. def forward(self, inputs):
  160. y = self.conv(inputs)
  161. y = self.pool2d_max(y)
  162. for bottleneck_block in self.bottleneck_block_list:
  163. y = bottleneck_block(y)
  164. y = self.pool2d_avg(y)
  165. y = fluid.layers.reshape(y, [y.shape[0], -1])
  166. y = self.out(y)
  167. return y
  1. with fluid.dygraph.guard():
  2. model = ResNet("ResNet")
  3. train(model)

通过运行结果可以发现,使用ResNet在眼疾筛查数据集iChallenge-PM上,loss能有效的下降,经过5个epoch的训练,在验证集上的准确率可以达到95%左右。

小结

在这一节里,给读者介绍了几种经典的图像分类模型,分别是LeNet, AlexNet, VGG, GoogLeNet和ResNet,并将它们应用到眼疾筛查数据集上。除了LeNet不适合大尺寸的图像分类问题之外,其它几个模型在此数据集上损失函数都能显著下降,在验证集上的预测精度在90%左右。如果读者有兴趣的话,可以进一步调整学习率和训练轮数等超参数,观察是否能够得到更高的精度。

参考文献

[1] Yann LeCun, Léon Bottou, Yoshua Bengio, and Patrick Haffner. Gradient-based learn- ing applied to document recognition. Proc. of the IEEE, 86(11):2278–2324, 1998

[2] Alex Krizhevsky, Ilya Sutskever, and Geoffrey E. Hinton. Imagenet classification with deep convolutional neural networks. In Advances in Neural Information Processing Systems, pages 1097–1105, 2012.

[3] Karen Simonyan and Andrew Zisserman. Very deep convolutional networks for large-scale image recognition. arXiv preprint arXiv:1409.1556, 2014b.

[4]Christian Szegedy, Wei Liu, Yangqing Jia, Pierre Sermanet, Scott Reed, Dragomir Anguelov, Dumitru Erhan, Vincent Vanhoucke, and Andrew Rabinovich. Going deeper with convolu- tions. In Proc. of the IEEE Conference on Computer Vision and Pattern Recognition, pages 1–9, 2015.

[5] Kaiming He, Xiangyu Zhang, Shaoqing Ren, and Jian Sun. Deep residual learning for im- age recognition. In Proc. of the IEEE Conference on Computer Vision and Pattern Recognition, pages 770–778, 2016a.

作业

1、如果将LeNet中中间层的激活函数Sigmoid换成ReLU,在眼底筛查数据集上将会得到什么样的结果?Loss是否能收敛,ReLU和Sigmoid之间的区别是引起结果不同的原因吗?请发表你的观点