1. Предварительная работа
В этой статье будет использоватьсяResNet-50
Реализовать распознавание и классификацию изображений птиц
моя среда:
- Язык: Python3.6.5
- Компилятор: Jupyter Notebook
- Среда глубокого обучения: TensorFlow2.4.1
- Адрес данных:【Портал】
? Из рубрики:【100 случаев глубокого обучения】
1. Настройте графический процессор
Если вы используете ЦП, вы можете закомментировать эту часть кода.
import tensorflow as tf
gpus = tf.config.list_physical_devices("GPU")
if gpus:
tf.config.experimental.set_memory_growth(gpus[0], True) #设置GPU显存用量按需使用
tf.config.set_visible_devices([gpus[0]],"GPU")
2. Импорт данных
import matplotlib.pyplot as plt
# 支持中文
plt.rcParams['font.sans-serif'] = ['SimHei'] # 用来正常显示中文标签
plt.rcParams['axes.unicode_minus'] = False # 用来正常显示负号
import os,PIL
# 设置随机种子尽可能使结果可以重现
import numpy as np
np.random.seed(1)
# 设置随机种子尽可能使结果可以重现
import tensorflow as tf
tf.random.set_seed(1)
from tensorflow import keras
from tensorflow.keras import layers,models
import pathlib
data_dir = "D:/jupyter notebook/DL-100-days/datasets/bird_photos"
data_dir = pathlib.Path(data_dir)
3. Просмотр данных
image_count = len(list(data_dir.glob('*/*')))
print("图片总数为:",image_count)
图片总数为: 565
2. Предварительная обработка данных
папка | количество |
---|---|
Bananaquit | 166 листов |
Black Throated Bushtiti | 111 листов |
Black skimmer | 122 листа |
Cockatoo | 166 листов |
1. Загрузите данные
использоватьimage_dataset_from_directory
метод загрузки данных с диска вtf.data.Dataset
середина
batch_size = 8
img_height = 224
img_width = 224
Студенты, чья версия TensorFlow 2.2.0, могут столкнуться сmodule 'tensorflow.keras.preprocessing' has no attribute 'image_dataset_from_directory'
Если сообщается об ошибке, обновите TensorFlow, и все будет в порядке.
"""
关于image_dataset_from_directory()的详细介绍可以参考文章:https://mtyjkh.blog.csdn.net/article/details/117018789
"""
train_ds = tf.keras.preprocessing.image_dataset_from_directory(
data_dir,
validation_split=0.2,
subset="training",
seed=123,
image_size=(img_height, img_width),
batch_size=batch_size)
Found 565 files belonging to 4 classes.
Using 452 files for training.
"""
关于image_dataset_from_directory()的详细介绍可以参考文章:https://mtyjkh.blog.csdn.net/article/details/117018789
"""
val_ds = tf.keras.preprocessing.image_dataset_from_directory(
data_dir,
validation_split=0.2,
subset="validation",
seed=123,
image_size=(img_height, img_width),
batch_size=batch_size)
Found 565 files belonging to 4 classes.
Using 113 files for validation.
Мы можем вывести метки набора данных через class_names. Метки будут соответствовать в алфавитном порядке именам каталогов.
class_names = train_ds.class_names
print(class_names)
['Bananaquit', 'Black Throated Bushtiti', 'Black skimmer', 'Cockatoo']
2. Визуализируйте данные
plt.figure(figsize=(10, 5)) # 图形的宽为10高为5
plt.suptitle("微信公众号:K同学啊")
for images, labels in train_ds.take(1):
for i in range(8):
ax = plt.subplot(2, 4, i + 1)
plt.imshow(images[i].numpy().astype("uint8"))
plt.title(class_names[labels[i]])
plt.axis("off")
plt.imshow(images[1].numpy().astype("uint8"))
3. Проверьте данные еще раз
for image_batch, labels_batch in train_ds:
print(image_batch.shape)
print(labels_batch.shape)
break
(8, 224, 224, 3)
(8,)
-
Image_batch
является тензором формы (8, 224, 224, 3). Это пакет из 8 изображений размером 240x240x3 (последнее измерение относится к цветовому каналу RGB). -
Label_batch
— тензор формы (8,), эти метки соответствуют 8 изображениям
4. Настройте набор данных
- shuffle(): скремблировать данные Подробное описание этой функции см. в:zhuanlan.zhihu.com/p/42417456
- prefetch(): Предварительная выборка данных и ускорение операции.Подробности см. в двух моих предыдущих статьях, которые объясняются в ней.
- cache(): кэшируйте набор данных в память, чтобы ускорить операцию.
AUTOTUNE = tf.data.AUTOTUNE
train_ds = train_ds.cache().shuffle(1000).prefetch(buffer_size=AUTOTUNE)
val_ds = val_ds.cache().prefetch(buffer_size=AUTOTUNE)
Если сообщаетсяAttributeError: module 'tensorflow._api.v2.data' has no attribute 'AUTOTUNE'
ошибка, будетAUTOTUNE = tf.data.AUTOTUNE
заменитьAUTOTUNE = tf.data.experimental.AUTOTUNE
3. Введение в остаточную сеть (ResNet)
1. Что решает остаточная сеть
Остаточная сеть предназначена для решения проблемы деградации сети, вызванной слишком большим количеством скрытых слоев нейронной сети. Проблема деградации означает, что когда в сети больше скрытых слоев, точность сети достигает насыщения, а затем резко ухудшается, и эта деградация не вызвана переоснащением.
расширять:«Две темные тучи» глубоких нейронных сетей
- Градиентное рассеивание/взрыв
Проще говоря, сеть слишком глубокая, что затруднит сходимость обучения модели. Эта проблема может быть эффективно решена методами стандартной инициализации и регуляризации промежуточного слоя. (Полезно знать это на данном этапе)
- деградация сети
По мере увеличения глубины сети производительность сети сначала постепенно увеличивается до насыщения, а затем быстро снижается, причем эта деградация не вызвана переоснащением.
2. Введение в ResNet-50
ResNet-50 имеет два основных блока, названныхConv Block
иIdentity Block
Структура конв. блока:
Структура блока идентификации:
Общая структура ResNet-50:
В-четвертых, постройте сетевую модель ResNet-50.
В центре внимания этой статьи находится следующее: вы можете попробовать собрать ResNet-50 самостоятельно в соответствии с тремя приведенными выше картинками.
from keras import layers
from keras.layers import Input,Activation,BatchNormalization,Flatten
from keras.layers import Dense,Conv2D,MaxPooling2D,ZeroPadding2D,AveragePooling2D
from keras.models import Model
def identity_block(input_tensor, kernel_size, filters, stage, block):
filters1, filters2, filters3 = filters
name_base = str(stage) + block + '_identity_block_'
x = Conv2D(filters1, (1, 1), name=name_base + 'conv1')(input_tensor)
x = BatchNormalization(name=name_base + 'bn1')(x)
x = Activation('relu', name=name_base + 'relu1')(x)
x = Conv2D(filters2, kernel_size,padding='same', name=name_base + 'conv2')(x)
x = BatchNormalization(name=name_base + 'bn2')(x)
x = Activation('relu', name=name_base + 'relu2')(x)
x = Conv2D(filters3, (1, 1), name=name_base + 'conv3')(x)
x = BatchNormalization(name=name_base + 'bn3')(x)
x = layers.add([x, input_tensor] ,name=name_base + 'add')
x = Activation('relu', name=name_base + 'relu4')(x)
return x
def conv_block(input_tensor, kernel_size, filters, stage, block, strides=(2, 2)):
filters1, filters2, filters3 = filters
res_name_base = str(stage) + block + '_conv_block_res_'
name_base = str(stage) + block + '_conv_block_'
x = Conv2D(filters1, (1, 1), strides=strides, name=name_base + 'conv1')(input_tensor)
x = BatchNormalization(name=name_base + 'bn1')(x)
x = Activation('relu', name=name_base + 'relu1')(x)
x = Conv2D(filters2, kernel_size, padding='same', name=name_base + 'conv2')(x)
x = BatchNormalization(name=name_base + 'bn2')(x)
x = Activation('relu', name=name_base + 'relu2')(x)
x = Conv2D(filters3, (1, 1), name=name_base + 'conv3')(x)
x = BatchNormalization(name=name_base + 'bn3')(x)
shortcut = Conv2D(filters3, (1, 1), strides=strides, name=res_name_base + 'conv')(input_tensor)
shortcut = BatchNormalization(name=res_name_base + 'bn')(shortcut)
x = layers.add([x, shortcut], name=name_base+'add')
x = Activation('relu', name=name_base+'relu4')(x)
return x
def ResNet50(input_shape=[224,224,3],classes=1000):
img_input = Input(shape=input_shape)
x = ZeroPadding2D((3, 3))(img_input)
x = Conv2D(64, (7, 7), strides=(2, 2), name='conv1')(x)
x = BatchNormalization(name='bn_conv1')(x)
x = Activation('relu')(x)
x = MaxPooling2D((3, 3), strides=(2, 2))(x)
x = conv_block(x, 3, [64, 64, 256], stage=2, block='a', strides=(1, 1))
x = identity_block(x, 3, [64, 64, 256], stage=2, block='b')
x = identity_block(x, 3, [64, 64, 256], stage=2, block='c')
x = conv_block(x, 3, [128, 128, 512], stage=3, block='a')
x = identity_block(x, 3, [128, 128, 512], stage=3, block='b')
x = identity_block(x, 3, [128, 128, 512], stage=3, block='c')
x = identity_block(x, 3, [128, 128, 512], stage=3, block='d')
x = conv_block(x, 3, [256, 256, 1024], stage=4, block='a')
x = identity_block(x, 3, [256, 256, 1024], stage=4, block='b')
x = identity_block(x, 3, [256, 256, 1024], stage=4, block='c')
x = identity_block(x, 3, [256, 256, 1024], stage=4, block='d')
x = identity_block(x, 3, [256, 256, 1024], stage=4, block='e')
x = identity_block(x, 3, [256, 256, 1024], stage=4, block='f')
x = conv_block(x, 3, [512, 512, 2048], stage=5, block='a')
x = identity_block(x, 3, [512, 512, 2048], stage=5, block='b')
x = identity_block(x, 3, [512, 512, 2048], stage=5, block='c')
x = AveragePooling2D((7, 7), name='avg_pool')(x)
x = Flatten()(x)
x = Dense(classes, activation='softmax', name='fc1000')(x)
model = Model(img_input, x, name='resnet50')
# 加载预训练模型
model.load_weights("resnet50_weights_tf_dim_ordering_tf_kernels.h5")
return model
model = ResNet50()
model.summary()
Model: "resnet50"
__________________________________________________________________________________________________
Layer (type) Output Shape Param # Connected to
==================================================================================================
input_1 (InputLayer) [(None, 224, 224, 3) 0
__________________________________________________________________________________________________
zero_padding2d (ZeroPadding2D) (None, 230, 230, 3) 0 input_1[0][0]
__________________________________________________________________________________________________
conv1 (Conv2D) (None, 112, 112, 64) 9472 zero_padding2d[0][0]
__________________________________________________________________________________________________
bn_conv1 (BatchNormalization) (None, 112, 112, 64) 256 conv1[0][0]
__________________________________________________________________________________________________
activation (Activation) (None, 112, 112, 64) 0 bn_conv1[0][0]
__________________________________________________________________________________________________
max_pooling2d (MaxPooling2D) (None, 55, 55, 64) 0 activation[0][0]
__________________________________________________________________________________________________
2a_conv_block_conv1 (Conv2D) (None, 55, 55, 64) 4160 max_pooling2d[0][0]
__________________________________________________________________________________________________
2a_conv_block_bn1 (BatchNormali (None, 55, 55, 64) 256 2a_conv_block_conv1[0][0]
__________________________________________________________________________________________________
2a_conv_block_relu1 (Activation (None, 55, 55, 64) 0 2a_conv_block_bn1[0][0]
__________________________________________________________________________________________________
2a_conv_block_conv2 (Conv2D) (None, 55, 55, 64) 36928 2a_conv_block_relu1[0][0]
__________________________________________________________________________________________________
2a_conv_block_bn2 (BatchNormali (None, 55, 55, 64) 256 2a_conv_block_conv2[0][0]
__________________________________________________________________________________________________
2a_conv_block_relu2 (Activation (None, 55, 55, 64) 0 2a_conv_block_bn2[0][0]
__________________________________________________________________________________________________
2a_conv_block_conv3 (Conv2D) (None, 55, 55, 256) 16640 2a_conv_block_relu2[0][0]
__________________________________________________________________________________________________
2a_conv_block_res_conv (Conv2D) (None, 55, 55, 256) 16640 max_pooling2d[0][0]
__________________________________________________________________________________________________
2a_conv_block_bn3 (BatchNormali (None, 55, 55, 256) 1024 2a_conv_block_conv3[0][0]
__________________________________________________________________________________________________
2a_conv_block_res_bn (BatchNorm (None, 55, 55, 256) 1024 2a_conv_block_res_conv[0][0]
__________________________________________________________________________________________________
2a_conv_block_add (Add) (None, 55, 55, 256) 0 2a_conv_block_bn3[0][0]
2a_conv_block_res_bn[0][0]
__________________________________________________________________________________________________
2a_conv_block_relu4 (Activation (None, 55, 55, 256) 0 2a_conv_block_add[0][0]
__________________________________________________________________________________________________
2b_identity_block_conv1 (Conv2D (None, 55, 55, 64) 16448 2a_conv_block_relu4[0][0]
__________________________________________________________________________________________________
2b_identity_block_bn1 (BatchNor (None, 55, 55, 64) 256 2b_identity_block_conv1[0][0]
=============================================================
此处省略了若干行,此处省略了若干行,此处省略了若干行
=============================================================
__________________________________________________________________________________________________
5c_identity_block_relu2 (Activa (None, 7, 7, 512) 0 5c_identity_block_bn2[0][0]
__________________________________________________________________________________________________
5c_identity_block_conv3 (Conv2D (None, 7, 7, 2048) 1050624 5c_identity_block_relu2[0][0]
__________________________________________________________________________________________________
5c_identity_block_bn3 (BatchNor (None, 7, 7, 2048) 8192 5c_identity_block_conv3[0][0]
__________________________________________________________________________________________________
5c_identity_block_add (Add) (None, 7, 7, 2048) 0 5c_identity_block_bn3[0][0]
5b_identity_block_relu4[0][0]
__________________________________________________________________________________________________
5c_identity_block_relu4 (Activa (None, 7, 7, 2048) 0 5c_identity_block_add[0][0]
__________________________________________________________________________________________________
avg_pool (AveragePooling2D) (None, 1, 1, 2048) 0 5c_identity_block_relu4[0][0]
__________________________________________________________________________________________________
flatten (Flatten) (None, 2048) 0 avg_pool[0][0]
__________________________________________________________________________________________________
fc1000 (Dense) (None, 1000) 2049000 flatten[0][0]
==================================================================================================
Total params: 25,636,712
Trainable params: 25,583,592
Non-trainable params: 53,120
__________________________________________________________________________________________________
5. Скомпилируйте
Прежде чем модель будет готова к обучению, необходимо выполнить еще несколько настроек. На этапе компиляции модели было добавлено следующее:
- Функция потерь (loss): используется для измерения точности модели во время обучения.
- Оптимизатор: определяет, как модель обновляется на основе данных, которые она видит, и собственной функции потерь.
- Метрики: используются для отслеживания этапов обучения и тестирования. В следующих примерах используется точность, которая представляет собой соотношение правильно классифицированных изображений.
# 设置优化器,我这里改变了学习率。
opt = tf.keras.optimizers.Adam(learning_rate=1e-7)
model.compile(optimizer="adam",
loss='sparse_categorical_crossentropy',
metrics=['accuracy'])
6. Обучение модели
epochs = 10
history = model.fit(
train_ds,
validation_data=val_ds,
epochs=epochs
)
Epoch 1/10
57/57 [==============================] - 12s 86ms/step - loss: 2.4313 - accuracy: 0.6548 - val_loss: 213.7383 - val_accuracy: 0.3186
Epoch 2/10
57/57 [==============================] - 3s 52ms/step - loss: 0.4293 - accuracy: 0.8557 - val_loss: 9.0470 - val_accuracy: 0.2566
Epoch 3/10
57/57 [==============================] - 3s 52ms/step - loss: 0.2309 - accuracy: 0.9183 - val_loss: 1.4181 - val_accuracy: 0.7080
Epoch 4/10
57/57 [==============================] - 3s 53ms/step - loss: 0.1721 - accuracy: 0.9535 - val_loss: 2.5627 - val_accuracy: 0.6726
Epoch 5/10
57/57 [==============================] - 3s 53ms/step - loss: 0.0795 - accuracy: 0.9701 - val_loss: 0.2747 - val_accuracy: 0.8938
Epoch 6/10
57/57 [==============================] - 3s 52ms/step - loss: 0.0435 - accuracy: 0.9899 - val_loss: 0.1483 - val_accuracy: 0.9381
Epoch 7/10
57/57 [==============================] - 3s 52ms/step - loss: 0.0308 - accuracy: 0.9970 - val_loss: 0.1705 - val_accuracy: 0.9381
Epoch 8/10
57/57 [==============================] - 3s 52ms/step - loss: 0.0019 - accuracy: 1.0000 - val_loss: 0.0674 - val_accuracy: 0.9735
Epoch 9/10
57/57 [==============================] - 3s 52ms/step - loss: 8.2391e-04 - accuracy: 1.0000 - val_loss: 0.0720 - val_accuracy: 0.9735
Epoch 10/10
57/57 [==============================] - 3s 52ms/step - loss: 6.0079e-04 - accuracy: 1.0000 - val_loss: 0.0762 - val_accuracy: 0.9646
7. Оценка модели
acc = history.history['accuracy']
val_acc = history.history['val_accuracy']
loss = history.history['loss']
val_loss = history.history['val_loss']
epochs_range = range(epochs)
plt.figure(figsize=(12, 4))
plt.subplot(1, 2, 1)
plt.suptitle("微信公众号:K同学啊")
plt.plot(epochs_range, acc, label='Training Accuracy')
plt.plot(epochs_range, val_acc, label='Validation Accuracy')
plt.legend(loc='lower right')
plt.title('Training and Validation Accuracy')
plt.subplot(1, 2, 2)
plt.plot(epochs_range, loss, label='Training Loss')
plt.plot(epochs_range, val_loss, label='Validation Loss')
plt.legend(loc='upper right')
plt.title('Training and Validation Loss')
plt.show()
Восемь, сохраните и загрузите модель
Это самый простой способ сохранять и загружать модели.
# 保存模型
model.save('model/my_model.h5')
# 加载模型
new_model = keras.models.load_model('model/my_model.h5')
9. Прогноз
# 采用加载的模型(new_model)来看预测结果
plt.figure(figsize=(10, 5)) # 图形的宽为10高为5
plt.suptitle("微信公众号:K同学啊")
for images, labels in val_ds.take(1):
for i in range(8):
ax = plt.subplot(2, 4, i + 1)
# 显示图片
plt.imshow(images[i].numpy().astype("uint8"))
# 需要给图片增加一个维度
img_array = tf.expand_dims(images[i], 0)
# 使用模型预测图片中的人物
predictions = new_model.predict(img_array)
plt.title(class_names[np.argmax(predictions)])
plt.axis("off")
? Из рубрики:【100 случаев глубокого обучения】
Если вы считаете, что эта статья полезна для вас, помнитеНажмите, чтобы подписаться, поставить лайк, добавить в избранное