Метод Tensorflow Finetune

машинное обучение глубокое обучение Нейронные сети

Недавно я сделал дипломную работу и хочу выполнить точную настройку непосредственно на модели предварительной подготовки Используемые фреймворки: tensorflow и keras. Итак, я поискал и обнаружил, что метод Finetune keras очень прост (представлен вкратце позже), но официальный сайт tensorflow тоже видит меня в каше, а метод от google не подходит.Во-первых, невозможно найти model_weights.,Во-вторых, нет описания способа добавления и обучения новых слоев в будущем.. Здесь я проворочался целый день, закончил со слезами и оставил заметку о прочтении, надеясь помочь людям с такими же потребностями.

------------------------------------------------ Великолепная сегментация Нить------------------------------------------------- -------

Роль предварительно обученных весов:

  • предсказывать
  • Извлечение признаков
  • тонкая настройка

Процесс тонкой настройки:

  • Построить структуру графа, перехватить целевой тензор, добавить новый слой
  • Загрузить целевые тензорные веса
  • тренировать новый слой
  • Глобальная тонкая настройка

1. Keras finetune

Модели глубокого обучения с предварительно обученными весами представлены в модуле приложений Keras.

Модуль автоматически проверит локальный~/.keras/models/Если каталог содержит нужный вес, он будет загружен автоматически, если нет, загрузка на ноутбук будет занимать ресурсы потока ноутбука, что неудобно, поэтому вы также можете вручную получить его.

приложения модуля приложения keras, возьмем MobileNet в качестве примера для иллюстрации:

#-----------------------------------构建模型------------------------------------
from keras.applications.mobilenet import MobileNet
from keras.layers import Input, Reshape, AvgPool2D,\
        Dropout, Conv2D, Softmax, BatchNormalization, Activation
from keras import Model

## 加载预训练权重,输入大小可以设定,include_top表示是否包括顶层的全连接层
base_model = MobileNet(input_shape=(128,128,3), include_top=False)

## 添加新层,get_layer方法可以根据层名返回该层,output用于返回该层的输出张量tensor
with tf.name_scope("output"):
    x = base_model.get_layer("conv_dw_6_relu").output
    x = Conv2D(256,kernel_size=(3,3))(x)
    x = Activation("relu")(x)
    x = AvgPool2D(pool_size=(5,5))(x)
    x = Dropout(rate=0.5)(x)
    x = Conv2D(10,kernel_size=(1,1),)(x)
    predictions = Reshape((10,))(x)

## finetune模型
model = Model(inputs=base_model.input, outputs=predictions)

#-------------------------------------训练新层-----------------------------------
## 冻结原始层位,在编译后生效
for layer in base_model.layers:
    layer.trainable = False

## 设定优化方法,并编译
sgd = keras.optimizers.SGD(lr=0.01)
model.compile(optimizer=sgd,loss="categorical_crossentropy")
‘’‘可选:记录模型训练过程、数据写入tensorboard
callback = [keras.callbacks.ModelCheckpoint(filepath="./vibration_keras/checkpoints",monitor="val_loss"),
           keras.callbacks.TensorBoard(log_dir="./vibration_keras/logs",histogram_freq=1,write_grads=True)]
’‘’

## 训练
model.fit(...)

#--------------------------------------全局微调-----------------------------------
## 设定各层是否用于训练,编译后生效
for layer in model.layers[:80]:
    layer.trainable = False
for layer in model.layers[80:]:
    layer.trainable = True

# 设定优化方法,并编译
sgd = keras.optimizer.SGD(lr=0.0001)
model.compile(optimizer=sgd, loss="categorical_crossentropy")

## 训练
model.fit(...)

Как получить имя каждого слоя:

Получить имя каждого слоя

2. Tensorflow finetune

Существует 3 метода тонкой настройки тензорного потока:

  • использоватьtf-slimВстроенная сетевая структура и веса настраиваются вручную
  • использоватьtf-slimкоторый предоставилtrain_image_classifier.pyСборка автоматизации скрипта, конкретный методздесь
  • использоватьtf.keras, процесс такой же, как keras

Первый метод, описанный выше, в основном представлен здесь, и меры предосторожности следующие:

  • tensorflow/models был удален из основного фреймворка tf после версии 1.0, и его необходимо загрузить вручную, расположение здесьtensorflow/models, вы можете использовать git clone для загрузки в локальный каталог, используйте следующую команду при использованииВременно добавлен в путь поиска pythonвнутри
import sys
sys.path.append("./models/research/slim")

2.1 Метод построения модели

В tensorflow есть 3 метода восстановления структуры и весов модели из файла контрольной точки.После восстановления перечисленных здесь моделей расчет прямой деривации может быть непосредственно выполнен для прогнозного анализа.

1) Загрузите структуру графика напрямую, а затем загрузите веса

# import_meta_graph可以直接从meta文件中加载图结构
saver = tf.train.import_meta_graph(os.path.join(model_path,r"resnet_v2/model.ckpt-258931.meta"))

# allow_soft_placement自动选择设备
with tf.Session(config=tf.ConfigProto(allow_soft_placement=True)) as sess:
    # latest_checkpoint检查checkpoint检查点文件,查找最新的模型
    # restore恢复图权重
    saver.restore(sess,tf.train.latest_checkpoint(r"./model/resnet_v2"))
    graph = sess.graph
    # get_tensor_by_name通过张量名称获取张量
    print(sess.run(graph.get_tensor_by_name("resnet_model/conv2d/kernel:0")))

2) Сначала постройте структуру графика, затем загрузите веса

# 临时添加slim到python搜索路径
import sys
sys.path.append("./models/research/slim")

# 导入mobilenet_v2
from nets.mobilenet import mobilenet_v2
# 重置图 
tf.reset_default_graph()

# 导入mobilenet,先构建图结构。
‘’‘加载完毕后,tf.get_default_graph()中包含了mobilenet计算图结构,
可以使用tf.get_collection(tf.GraphKeys.TRAINABLE_VARIABLES)对比reset_graph前后的差异
’‘’
images = tf.placeholder(tf.float32,(None,224,224,3))
with tf.contrib.slim.arg_scope(mobilenet_v2.training_scope(is_training=False)):
    logits, endpoints = mobilenet_v2.mobilenet(images,depth_multiplier=1.4)

# 定义saver类,用于恢复图权重
saver = tf.train.Saver()
with tf.Session() as sess:
    # latest_checkpoint检查checkpoint检查点文件,查找最新的模型
    # restore恢复图权重
    saver.restore(sess,tf.train.latest_checkpoint("./model_ckpt/mobilenet_v2"))
    # get_tensor_by_name通过张量名称获取张量
    print(sess.run(tf.get_default_graph().get_tensor_by_name("MobilenetV2/Conv/weights:0")).shape)

вывести имя узла в структуре вычислительного графа,За именем тензора должно следовать число, например: 0, который представляет первый вывод вычислительного узла:

Все имена обучаемых переменных

3) frozen inference

Файл pb помещает значения переменных и всю структуру графа расчета в один файл и преобразует переменные и значения в константы через convert_variable_to_constants для сохранения.Когда модель тестируется, ввод нужно только перенаправить в выходной слой.

# 读取保存的pb文件,并解析成对应的GraphDef Protocol Buffer
gd =  tf.GraphDef.FromString(open('./model_ckpt/mobilenet_v2/mobilenet_v2_1.4_224_frozen.pb',"rb").read())
# import_graph_def将graphdef中保存的图加载到当前图中,return_elements返回指定张量
inp, predictions = \
tf.import_graph_def(gd,return_elements=["input:0","MobilenetV2/Predictions/Reshape_1:0"])

# 此时的计算图可以直接用于预测
# 拉取一张图片 !wget https://upload.wikimedia.org/wikipedia/commons/f/fe/Giant_Panda_in_Beijing_Zoo_1.JPG -O panda.jpg
from PIL import Image
img = np.array(Image.open('panda.jpg').resize((224, 224))).astype(np.float) / 128 - 1
# inp是需要feed的输入,predictions是需要输出的预测结构
with tf.Session(graph=inp.graph) as sess:
    x = sess.run(predictions,feed_dict={inp:img.reshape(1,224,224,3)})

2.2 Процесс тонкой настройки

  • 1) Постройте структуру графа, перехватите целевой тензор и добавьте новый слой
  • 2) Загрузите целевые тензорные веса
  • 3) Обучить новый слой
  • 4) Глобальная тонкая настройка

1) Постройте структуру графа, перехватите целевой тензор и добавьте новый слой

Структура графа на этом шаге создается с помощью "Сначала постройте структуру графика, затем загрузите веса"Структура вычислительного графа мобильной сети, полученная данным методом.

tf.reset_default_graph()
# 构建计算图
images = tf.placeholder(tf.float32,(None,224,224,3))
with tf.contrib.slim.arg_scope(mobilenet_v2.training_scope(is_training=False)):
    logits, endpoints = mobilenet_v2.mobilenet(images,depth_multiplier=1.4)

# 获取目标张量,添加新层
with tf.variable_scope("finetune_layers"):
    # 获取目标张量,取出mobilenet中指定层的张量
    mobilenet_tensor = tf.get_default_graph().get_tensor_by_name("MobilenetV2/expanded_conv_14/output:0")
    # 将张量向新层传递
    x = tf.layers.Conv2D(filters=256,kernel_size=3,name="conv2d_1")(mobilenet_tensor)
    # 观察新层权重是否更新 tf.summary.histogram("conv2d_1",x)
    x = tf.nn.relu(x,name="relu_1")
    x = tf.layers.Conv2D(filters=256,kernel_size=3,name="conv2d_2")(x)
    x = tf.layers.Conv2D(10,3,name="conv2d_3")(x)
    predictions = tf.reshape(x, (-1,10))

Структура вычислительного графа:

тонкая настройка структуры сети

Красное поле — это структура сети Mobilenet Второй фиолетовый узел сверху вниз — это узел «MobilenetV2/expanded_conv_14/output», который напрямую связан с Finetune_layers.

2) Загрузите целевые веса и обучите новый слой

# one-hot编码
def to_categorical(data, nums):
    return np.eye(nums)[data]
# 随机生成数据
x_train = np.random.random(size=(141,224,224,3))
y_train = to_categorical(label_fake,10)

# 训练条件配置
## label占位符
y_label = tf.placeholder(tf.int32, (None,10))
## 收集变量作用域finetune_layers内的变量,仅更新添加层的权重
train_var = tf.get_collection(tf.GraphKeys.TRAINABLE_VARIABLES,scope="finetune_layers")
## 定义loss
loss = tf.nn.softmax_cross_entropy_with_logits_v2(labels=y_label,logits=predictions)
## 定义优化方法,用var_list指定需要更新的权重,此时仅更新train_var权重
train_step = tf.train.GradientDescentOptimizer(0.1).minimize(loss,var_list=train_var)
## 观察新层权重是否更新 
tf.summary.histogram("mobilenet_conv8",tf.get_default_graph().get_tensor_by_name('MobilenetV2/expanded_conv_8/depthwise/depthwise_weights:0'))
tf.summary.histogram("mobilenet_conv9",tf.get_default_graph().get_tensor_by_name('MobilenetV2/expanded_conv_9/depthwise/depthwise_weights:0'))

## 合并所有summary
merge_all = tf.summary.merge_all()

## 设定迭代次数和批量大学
epochs = 10
batch_size = 16

# 获取指定变量列表var_list的函数
def get_var_list(target_tensor=None):
    '''获取指定变量列表var_list的函数'''
    if target_tensor==None:
        target_tensor = r"MobilenetV2/expanded_conv_14/output:0"
    target = target_tensor.split("/")[1]
    all_list = []
    all_var = []
    # 遍历所有变量,node.name得到变量名称
    # 不使用tf.trainable_variables(),因为batchnorm的moving_mean/variance不属于可训练变量
    for var in tf.global_variables():
        if var != []:
            all_list.append(var.name)
            all_var.append(var)
    try:
        all_list = list(map(lambda x:x.split("/")[1],all_list))
        # 查找对应变量作用域的索引
        ind = all_list[::-1].index(target)
        ind = len(all_list) -  ind - 1
        print(ind)
        del all_list
        return all_var[:ind+1]
    except:
        print("target_tensor is not exist!")

# 目标张量名称,要获取一个需要从文件中加载权重的变量列表var_list
target_tensor = "MobilenetV2/expanded_conv_14/output:0"
var_list = get_var_list(target_tensor)
saver = tf.train.Saver(var_list=var_list)

# 加载文件内的权重,并训练新层
with tf.Session(config=tf.ConfigProto(allow_soft_placement=True)) as sess:
    writer = tf.summary.FileWriter(r"./logs", sess.graph)
    ## 初始化参数:从文件加载权重 train_var使用初始化函数
    sess.run(tf.variables_initializer(var_list=train_var))
    saver.restore(sess,tf.train.latest_checkpoint("./model_ckpt/mobilenet_v2"))
    
    for i in range(2000):
        start = (i*batch_size) % x_train.shape[0]
        end = min(start+batch_size, x_train.shape[0])
        _, merge, losses = sess.run([train_step,merge_all,loss],\
                             feed_dict={images:x_train[start:end],\
                                        y_label:y_train[start:end]})
        if i%100==0:
           writer.add_summary(merge, i)

Соображения по инициализации веса:

1. Сначала используйте глобальную инициализацию tf.global_variables_initializer(), а затем используйте порядок saver.restore не может быть неправильным, иначе загруженные веса будут повторно инициализированы.

sess.run(tf.global_variables_initializer())
saver.restore(sess,tf.train.latest_checkpoint("./model_ckpt/mobilenet_v2"))

2. Сначала используйте saver.restore для загрузки весов из модели, а затем используйте tf.variable_initializaer() для инициализации указанного var_list, порядок может быть обратным.

saver.restore(sess,tf.train.latest_checkpoint("./model_ckpt/mobilenet_v2"))
sess.run(tf.variables_initializer(var_list=train_var))

3. Первые два метода также инициализируют переменные для бесполезных узлов и должны заранее выполнить операцию saver.restore, что означает, что требуются две операции save.restore, чтобы гарантировать, что процесс тонкой настройки не сообщит об ошибках. Теперь вы можете отфильтровать все переменные, которым необходимо загрузить веса из файла, чтобы сформировать var_list, а затем определить saver=tf.train.Saver(var_list) для выборочной загрузки переменных.

В приведенном выше коде используется третий метод, и разницу между тремя вышеуказанными методами инициализации можно хорошо понять. Посмотрите внимательно на скриншот ниже.

Различия, вызванные разными инициализациями

----------------------------------------------------------Обновление 20180817---- -----------------------------------------

1) Добавлено размышление о 3 методах инициализации переменных

2) Скорректировать логику статьи. 2.1 Построение модели в основном используется для вывода прямого распространения, 2.2 описывает полный процесс тонкой настройки.

Добро пожаловать на перепечатку ~ пожалуйста, укажите источник ~ спасибо ~