CNTK - в памяти и большие наборы данных

В этой главе мы узнаем, как работать с наборами данных в памяти и большими наборами данных в CNTK.

Обучение с небольшими наборами данных в памяти

Когда мы говорим о загрузке данных в тренер CNTK, может быть много способов, но это будет зависеть от размера набора данных и формата данных. Наборы данных могут быть небольшими в памяти или большими наборами данных.

В этом разделе мы собираемся работать с наборами данных в памяти. Для этого мы будем использовать следующие два фреймворка -

  • Numpy
  • Pandas

Использование массивов Numpy

Здесь мы будем работать со случайно сгенерированным набором данных на основе numpy в CNTK. В этом примере мы будем моделировать данные для задачи двоичной классификации. Предположим, у нас есть набор наблюдений с 4 функциями и мы хотим предсказать две возможные метки с помощью нашей модели глубокого обучения.

Пример реализации

Для этого сначала мы должны сгенерировать набор меток, содержащий горячее векторное представление меток, которые мы хотим спрогнозировать. Это можно сделать с помощью следующих шагов -

Step 1 - Импортировать numpy пакет следующим образом -

import numpy as np
num_samples = 20000

Step 2 - Затем сгенерируйте сопоставление меток с помощью np.eye функционируют следующим образом -

label_mapping = np.eye(2)

Step 3 - Теперь с помощью np.random.choice функция, соберите 20000 случайных выборок следующим образом -

y = label_mapping[np.random.choice(2,num_samples)].astype(np.float32)

Step 4 - Теперь, наконец, используя функцию np.random.random, сгенерируйте массив случайных значений с плавающей запятой следующим образом:

x = np.random.random(size=(num_samples, 4)).astype(np.float32)

После того, как мы сгенерируем массив случайных значений с плавающей запятой, нам нужно преобразовать их в 32-битные числа с плавающей запятой, чтобы его можно было сопоставить с форматом, ожидаемым CNTK. Чтобы сделать это, выполните следующие действия:

Step 5 - Импортируйте функции плотного и последовательного слоя из модуля cntk.layers следующим образом:

from cntk.layers import Dense, Sequential

Step 6- Теперь нам нужно импортировать функцию активации для слоев в сети. Давайте импортируемsigmoid как функция активации -

from cntk import input_variable, default_options
from cntk.ops import sigmoid

Step 7- Теперь нам нужно импортировать функцию потерь для обучения сети. Давайте импортируемbinary_cross_entropy как функция потерь -

from cntk.losses import binary_cross_entropy

Step 8- Далее нам нужно определить параметры сети по умолчанию. Здесь мы предоставимsigmoidфункция активации по умолчанию. Кроме того, создайте модель с помощью функции последовательного слоя следующим образом:

with default_options(activation=sigmoid):
model = Sequential([Dense(6),Dense(2)])

Step 9 - Затем инициализируйте input_variable с 4 входными функциями, которые служат входом для сети.

features = input_variable(4)

Step 10 - Теперь, чтобы завершить его, нам нужно подключить переменную features к NN.

z = model(features)

Итак, теперь у нас есть NN, с помощью следующих шагов давайте обучим его, используя набор данных в памяти -

Step 11 - Чтобы обучить эту NN, сначала нам нужно импортировать ученика из cntk.learnersмодуль. Мы импортируемsgd ученик следующим образом -

from cntk.learners import sgd

Step 12 - Наряду с этим импортировать ProgressPrinter из cntk.logging модуль тоже.

from cntk.logging import ProgressPrinter
progress_writer = ProgressPrinter(0)

Step 13 - Затем определите новую входную переменную для меток следующим образом -

labels = input_variable(2)

Step 14 - Затем, чтобы обучить модель NN, нам нужно определить потери, используя binary_cross_entropyфункция. Также укажите модель z и переменную меток.

loss = binary_cross_entropy(z, labels)

Step 15 - Затем инициализируйте sgd ученик следующим образом -

learner = sgd(z.parameters, lr=0.1)

Step 16- Наконец, вызовите метод поезда для функции потерь. Кроме того, предоставьте ему входные данные,sgd ученик и progress_printer.−

training_summary=loss.train((x,y),parameter_learners=[learner],callbacks=[progress_writer])

Полный пример реализации

import numpy as np
num_samples = 20000
label_mapping = np.eye(2)
y = label_mapping[np.random.choice(2,num_samples)].astype(np.float32)
x = np.random.random(size=(num_samples, 4)).astype(np.float32)
from cntk.layers import Dense, Sequential
from cntk import input_variable, default_options
from cntk.ops import sigmoid
from cntk.losses import binary_cross_entropy
with default_options(activation=sigmoid):
   model = Sequential([Dense(6),Dense(2)])
features = input_variable(4)
z = model(features)
from cntk.learners import sgd
from cntk.logging import ProgressPrinter
progress_writer = ProgressPrinter(0)
labels = input_variable(2)
loss = binary_cross_entropy(z, labels)
learner = sgd(z.parameters, lr=0.1)
training_summary=loss.train((x,y),parameter_learners=[learner],callbacks=[progress_writer])

Вывод

Build info:
     Built time: *** ** **** 21:40:10
     Last modified date: *** *** ** 21:08:46 2019
     Build type: Release
     Build target: CPU-only
     With ASGD: yes
     Math lib: mkl
     Build Branch: HEAD
     Build SHA1:ae9c9c7c5f9e6072cc9c94c254f816dbdc1c5be6 (modified)
     MPI distribution: Microsoft MPI
     MPI version: 7.0.12437.6
-------------------------------------------------------------------
average   since   average   since examples
loss      last    metric    last
------------------------------------------------------
Learning rate per minibatch: 0.1
1.52      1.52      0         0     32
1.51      1.51      0         0     96
1.48      1.46      0         0    224
1.45      1.42      0         0    480
1.42       1.4      0         0    992
1.41      1.39      0         0   2016
1.4       1.39      0         0   4064
1.39      1.39      0         0   8160
1.39      1.39      0         0  16352

Использование Pandas DataFrames

Массивы Numpy очень ограничены в том, что они могут содержать, и являются одним из самых основных способов хранения данных. Например, один n-мерный массив может содержать данные одного типа данных. Но с другой стороны, для многих реальных случаев нам нужна библиотека, которая может обрабатывать более одного типа данных в одном наборе данных.

Одна из библиотек Python под названием Pandas упрощает работу с такими наборами данных. Он вводит концепцию DataFrame (DF) и позволяет нам загружать наборы данных с диска, хранящиеся в различных форматах, как DF. Например, мы можем читать DF, хранящиеся в CSV, JSON, Excel и т. Д.

Вы можете изучить библиотеку Python Pandas более подробно на https://www.tutorialspoint.com/python_pandas/index.htm.

Пример реализации

В этом примере мы собираемся использовать пример классификации трех возможных видов цветов ириса на основе четырех свойств. Мы создали эту модель глубокого обучения и в предыдущих разделах. Модель выглядит следующим образом -

from cntk.layers import Dense, Sequential
from cntk import input_variable, default_options
from cntk.ops import sigmoid, log_softmax
from cntk.losses import binary_cross_entropy
model = Sequential([
Dense(4, activation=sigmoid),
Dense(3, activation=log_softmax)
])
features = input_variable(4)
z = model(features)

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

Далее мы будем использовать train метод и lossфункция для обучения сети. Для этого сначала мы должны загрузить и предварительно обработать набор данных радужной оболочки, чтобы он соответствовал ожидаемому макету и формату данных для NN. Это можно сделать с помощью следующих шагов -

Step 1 - Импортировать numpy и Pandas пакет следующим образом -

import numpy as np
import pandas as pd

Step 2 - Далее используйте read_csv функция для загрузки набора данных в память -

df_source = pd.read_csv(‘iris.csv’, names = [‘sepal_length’, ‘sepal_width’,
 ‘petal_length’, ‘petal_width’, ‘species’], index_col=False)

Step 3 - Теперь нам нужно создать словарь, который будет отображать метки в наборе данных с их соответствующим числовым представлением.

label_mapping = {‘Iris-Setosa’ : 0, ‘Iris-Versicolor’ : 1, ‘Iris-Virginica’ : 2}

Step 4 - Теперь, используя iloc индексатор на DataFrame, выберите первые четыре столбца следующим образом -

x = df_source.iloc[:, :4].values

Step 5−Далее нам нужно выбрать столбцы видов в качестве меток для набора данных. Это можно сделать следующим образом -

y = df_source[‘species’].values

Step 6 - Теперь нам нужно сопоставить метки в наборе данных, что можно сделать с помощью label_mapping. Также используйтеone_hot кодирование для преобразования их в массивы горячего кодирования.

y = np.array([one_hot(label_mapping[v], 3) for v in y])

Step 7 - Затем, чтобы использовать функции и сопоставленные метки с CNTK, нам нужно преобразовать их оба в числа с плавающей запятой -

x= x.astype(np.float32)
y= y.astype(np.float32)

Как мы знаем, метки хранятся в наборе данных в виде строк, и CNTK не может работать с этими строками. По этой причине ему нужны векторы с горячим кодированием, представляющие метки. Для этого мы можем определить функцию sayone_hot следующим образом -

def one_hot(index, length):
result = np.zeros(length)
result[index] = index
return result

Теперь у нас есть массив numpy в правильном формате, с помощью следующих шагов мы можем использовать их для обучения нашей модели:

Step 8- Во-первых, нам нужно импортировать функцию потерь для обучения сети. Давайте импортируемbinary_cross_entropy_with_softmax как функция потерь -

from cntk.losses import binary_cross_entropy_with_softmax

Step 9 - Чтобы обучить эту NN, нам также нужно импортировать ученика из cntk.learnersмодуль. Мы импортируемsgd ученик следующим образом -

from cntk.learners import sgd

Step 10 - Наряду с этим импортировать ProgressPrinter из cntk.logging модуль тоже.

from cntk.logging import ProgressPrinter
progress_writer = ProgressPrinter(0)

Step 11 - Затем определите новую входную переменную для меток следующим образом -

labels = input_variable(3)

Step 12 - Затем, чтобы обучить модель NN, нам нужно определить потери, используя binary_cross_entropy_with_softmaxфункция. Также укажите модель z и переменную меток.

loss = binary_cross_entropy_with_softmax (z, labels)

Step 13 - Затем инициализируйте sgd ученик следующим образом -

learner = sgd(z.parameters, 0.1)

Step 14- Наконец, вызовите метод поезда для функции потерь. Кроме того, предоставьте ему входные данные,sgd ученик и progress_printer.

training_summary=loss.train((x,y),parameter_learners=[learner],callbacks=
[progress_writer],minibatch_size=16,max_epochs=5)

Полный пример реализации

from cntk.layers import Dense, Sequential
from cntk import input_variable, default_options
from cntk.ops import sigmoid, log_softmax
from cntk.losses import binary_cross_entropy
model = Sequential([
Dense(4, activation=sigmoid),
Dense(3, activation=log_softmax)
])
features = input_variable(4)
z = model(features)
import numpy as np
import pandas as pd
df_source = pd.read_csv(‘iris.csv’, names = [‘sepal_length’, ‘sepal_width’, ‘petal_length’, ‘petal_width’, ‘species’], index_col=False)
label_mapping = {‘Iris-Setosa’ : 0, ‘Iris-Versicolor’ : 1, ‘Iris-Virginica’ : 2}
x = df_source.iloc[:, :4].values
y = df_source[‘species’].values
y = np.array([one_hot(label_mapping[v], 3) for v in y])
x= x.astype(np.float32)
y= y.astype(np.float32)
def one_hot(index, length):
result = np.zeros(length)
result[index] = index
return result
from cntk.losses import binary_cross_entropy_with_softmax
from cntk.learners import sgd
from cntk.logging import ProgressPrinter
progress_writer = ProgressPrinter(0)
labels = input_variable(3)
loss = binary_cross_entropy_with_softmax (z, labels)
learner = sgd(z.parameters, 0.1)
training_summary=loss.train((x,y),parameter_learners=[learner],callbacks=[progress_writer],minibatch_size=16,max_epochs=5)

Вывод

Build info:
     Built time: *** ** **** 21:40:10
     Last modified date: *** *** ** 21:08:46 2019
     Build type: Release
     Build target: CPU-only
     With ASGD: yes
     Math lib: mkl
     Build Branch: HEAD
     Build SHA1:ae9c9c7c5f9e6072cc9c94c254f816dbdc1c5be6 (modified)
     MPI distribution: Microsoft MPI
     MPI version: 7.0.12437.6
-------------------------------------------------------------------
average    since    average   since   examples
loss        last     metric   last
------------------------------------------------------
Learning rate per minibatch: 0.1
1.1         1.1        0       0      16
0.835     0.704        0       0      32
1.993      1.11        0       0      48
1.14       1.14        0       0     112
[………]

Обучение с большими наборами данных

В предыдущем разделе мы работали с небольшими наборами данных в памяти, используя Numpy и pandas, но не все наборы данных такие маленькие. Особенно большие наборы данных, содержащие изображения, видео, звуковые образцы.MinibatchSource- это компонент, который может загружать данные по частям, предоставляемый CNTK для работы с такими большими наборами данных. Некоторые особенностиMinibatchSource компоненты следующие -

  • MinibatchSource может предотвратить переобучение NN за счет автоматического рандомизации выборок, считываемых из источника данных.

  • Он имеет встроенный конвейер преобразования, который можно использовать для увеличения данных.

  • Он загружает данные в фоновый поток отдельно от процесса обучения.

В следующих разделах мы собираемся изучить, как использовать источник мини-пакета с данными нехватки памяти для работы с большими наборами данных. Мы также рассмотрим, как мы можем использовать его для обучения NN.

Создание экземпляра MinibatchSource

В предыдущем разделе мы использовали пример цветка ириса и работали с небольшим набором данных в памяти, используя Pandas DataFrames. Здесь мы заменим код, который использует данные из pandas DF, наMinibatchSource. Во-первых, нам нужно создать экземплярMinibatchSource с помощью следующих шагов -

Пример реализации

Step 1 - Во-первых, из cntk.io модуль импортирует компоненты для источника мини-пакетов следующим образом:

from cntk.io import StreamDef, StreamDefs, MinibatchSource, CTFDeserializer,
 INFINITY_REPEAT

Step 2 - Теперь, используя StreamDef class, создайте определение потока для меток.

labels_stream = StreamDef(field=’labels’, shape=3, is_sparse=False)

Step 3 - Затем создайте для чтения функций, поданных из входного файла, создайте еще один экземпляр StreamDef следующим образом.

feature_stream = StreamDef(field=’features’, shape=4, is_sparse=False)

Step 4 - Теперь нам нужно предоставить iris.ctf файл в качестве ввода и инициализировать deserializer следующим образом -

deserializer = CTFDeserializer(‘iris.ctf’, StreamDefs(labels=
label_stream, features=features_stream)

Step 5 - Наконец, нам нужно создать экземпляр minisourceBatch используя deserializer следующим образом -

Minibatch_source = MinibatchSource(deserializer, randomize=True)

Создание экземпляра MinibatchSource - полный пример реализации

from cntk.io import StreamDef, StreamDefs, MinibatchSource, CTFDeserializer, INFINITY_REPEAT
labels_stream = StreamDef(field=’labels’, shape=3, is_sparse=False)
feature_stream = StreamDef(field=’features’, shape=4, is_sparse=False)
deserializer = CTFDeserializer(‘iris.ctf’, StreamDefs(labels=label_stream, features=features_stream)
Minibatch_source = MinibatchSource(deserializer, randomize=True)

Создание файла MCTF

Как вы видели выше, мы берем данные из файла iris.ctf. Он имеет формат файла, называемый текстовым форматом CNTK (CTF). Обязательно создать файл CTF для получения данных дляMinibatchSourceэкземпляр, который мы создали выше. Давайте посмотрим, как мы можем создать файл CTF.

Пример реализации

Step 1 - Во-первых, нам нужно импортировать пакеты pandas и numpy следующим образом:

import pandas as pd
import numpy as np

Step 2- Затем нам нужно загрузить в память наш файл данных, т.е. iris.csv. Затем сохраните его вdf_source переменная.

df_source = pd.read_csv(‘iris.csv’, names = [‘sepal_length’, ‘sepal_width’, ‘petal_length’, ‘petal_width’, ‘species’], index_col=False)

Step 3 - Теперь, используя ilocindexer в качестве функций, возьмите содержимое первых четырех столбцов. Кроме того, используйте данные из столбца видов следующим образом:

features = df_source.iloc[: , :4].values
labels = df_source[‘species’].values

Step 4- Затем нам нужно создать соответствие между именем метки и ее числовым представлением. Это можно сделать, создавlabel_mapping следующим образом -

label_mapping = {‘Iris-Setosa’ : 0, ‘Iris-Versicolor’ : 1, ‘Iris-Virginica’ : 2}

Step 5 - Теперь преобразуйте метки в набор векторов с горячим кодированием следующим образом:

labels = [one_hot(label_mapping[v], 3) for v in labels]

Теперь, как и раньше, создайте служебную функцию с именем one_hotдля кодирования этикеток. Это можно сделать следующим образом -

def one_hot(index, length):
result = np.zeros(length)
result[index] = 1
return result

Поскольку мы загрузили и предварительно обработали данные, самое время сохранить их на диске в формате файла CTF. Мы можем сделать это с помощью следующего кода Python -

With open(‘iris.ctf’, ‘w’) as output_file:
for index in range(0, feature.shape[0]):
feature_values = ‘ ‘.join([str(x) for x in np.nditer(features[index])])
label_values = ‘ ‘.join([str(x) for x in np.nditer(labels[index])])
output_file.write(‘features {} | labels {} \n’.format(feature_values, label_values))

Создание файла MCTF - Полный пример реализации

import pandas as pd
import numpy as np
df_source = pd.read_csv(‘iris.csv’, names = [‘sepal_length’, ‘sepal_width’, ‘petal_length’, ‘petal_width’, ‘species’], index_col=False)
features = df_source.iloc[: , :4].values
labels = df_source[‘species’].values
label_mapping = {‘Iris-Setosa’ : 0, ‘Iris-Versicolor’ : 1, ‘Iris-Virginica’ : 2}
labels = [one_hot(label_mapping[v], 3) for v in labels]
def one_hot(index, length):
result = np.zeros(length)
result[index] = 1
return result
With open(‘iris.ctf’, ‘w’) as output_file:
for index in range(0, feature.shape[0]):
feature_values = ‘ ‘.join([str(x) for x in np.nditer(features[index])])
label_values = ‘ ‘.join([str(x) for x in np.nditer(labels[index])])
output_file.write(‘features {} | labels {} \n’.format(feature_values, label_values))

Подача данных

Как только вы создадите MinibatchSource,Например, нам нужно обучить его. Мы можем использовать ту же логику обучения, что и при работе с небольшими наборами данных в памяти. Здесь мы будем использоватьMinibatchSource instance в качестве входа для метода поезда для функции потерь следующим образом:

Пример реализации

Step 1 - Чтобы записать результат тренировки, сначала импортируйте ProgressPrinter из cntk.logging модуль следующим образом -

from cntk.logging import ProgressPrinter

Step 2 - Затем, чтобы настроить тренировку, импортируйте trainer и training_session из cntk.train модуль следующим образом -

from cntk.train import Trainer,

Step 3 - Теперь нам нужно определить некоторый набор констант, например minibatch_size, samples_per_epoch и num_epochs следующим образом -

minbatch_size = 16
samples_per_epoch = 150
num_epochs = 30

Step 4 - Затем, чтобы узнать, как CNTK читать данные во время обучения, нам нужно определить соответствие между входной переменной для сети и потоками в источнике мини-пакета.

input_map = {
     features: minibatch.source.streams.features,
     labels: minibatch.source.streams.features
}

Step 5 - Затем, чтобы записать результат тренировочного процесса, инициализируйте progress_printer переменная с новым ProgressPrinter пример следующим образом -

progress_writer = ProgressPrinter(0)

Step 6 - Наконец, нам нужно вызвать метод поезда для потери следующим образом -

train_history = loss.train(minibatch_source,
parameter_learners=[learner],
  model_inputs_to_streams=input_map,
callbacks=[progress_writer],
epoch_size=samples_per_epoch,
max_epochs=num_epochs)

Подача данных - полный пример реализации

from cntk.logging import ProgressPrinter
from cntk.train import Trainer, training_session
minbatch_size = 16
samples_per_epoch = 150
num_epochs = 30
input_map = {
   features: minibatch.source.streams.features,
   labels: minibatch.source.streams.features
}
progress_writer = ProgressPrinter(0)
train_history = loss.train(minibatch_source,
parameter_learners=[learner],
model_inputs_to_streams=input_map,
callbacks=[progress_writer],
epoch_size=samples_per_epoch,
max_epochs=num_epochs)

Вывод

-------------------------------------------------------------------
average   since   average   since  examples
loss      last     metric   last
------------------------------------------------------
Learning rate per minibatch: 0.1
1.21      1.21      0        0       32
1.15      0.12      0        0       96
[………]