CNTK - Evrişimli Sinir Ağı
Bu bölümde, CNTK'da bir Evrişimli Sinir Ağı'nın (CNN) nasıl kurulacağını inceleyelim.
Giriş
Evrişimli sinir ağları (CNN'ler) ayrıca öğrenilebilir ağırlıklara ve önyargılara sahip nöronlardan oluşur. Bu yüzden bu şekilde sıradan sinir ağları (NN'ler) gibiler.
Sıradan NN'lerin çalışmasını hatırlarsak, her nöron bir veya daha fazla girdi alır, ağırlıklı bir toplam alır ve son çıktıyı üretmek için bir aktivasyon fonksiyonundan geçer. Burada şu soru ortaya çıkıyor: CNN'ler ve sıradan NN'ler bu kadar çok benzerliğe sahipse, o zaman bu iki ağı birbirinden farklı kılan nedir?
Bunları farklı kılan, girdi verilerinin ve katman türlerinin işlenmesidir? Girdi verilerinin yapısı sıradan NN'de göz ardı edilir ve tüm veriler ağa beslenmeden önce 1 boyutlu diziye dönüştürülür.
Ancak Evrişimli Sinir Ağı mimarisi, görüntülerin 2D yapısını dikkate alabilir, işleyebilir ve görüntülere özgü özellikleri çıkarmasına izin verebilir. Ayrıca, CNN'ler, CNN'lerin ana yapı taşları olan bir veya daha fazla Evrişimli katmana ve havuz katmanına sahip olma avantajına sahiptir.
Bu katmanları, standart çok katmanlı NN'lerde olduğu gibi bir veya daha fazla tamamen bağlı katman izler. Dolayısıyla, CNN'i tamamen bağlı ağların özel bir durumu olarak düşünebiliriz.
Evrişimli Sinir Ağı (CNN) mimarisi
CNN mimarisi temelde 3 boyutlu, yani görüntü hacminin genişliği, yüksekliği ve derinliğini 3 boyutlu çıktı hacmine dönüştüren bir katman listesidir. Burada dikkat edilmesi gereken önemli bir nokta, mevcut katmandaki her nöronun, önceki katmandan gelen çıktının küçük bir yamasına bağlanmasıdır; bu, giriş görüntüsünün üzerine bir N * N filtre yerleştirmek gibidir.
Temelde kenarlar, köşe vb. Gibi özellikleri çıkaran özellik çıkarıcılar olan M filtreleri kullanır. Katmanlar aşağıdadır [INPUT-CONV-RELU-POOL-FC] Evrişimli sinir ağları (CNN'ler) oluşturmak için kullanılan
INPUT- Adından da anlaşılacağı gibi, bu katman ham piksel değerlerini tutar. Ham piksel değerleri, görüntünün verilerini olduğu gibi ifade eder. Örnek, INPUT [64 × 64 × 3], genişlik-64, yükseklik-64 ve derinlik-3 olan 3 kanallı bir RGB görüntüsüdür.
CONV- Hesaplamanın çoğu bu katmanda yapıldığı için bu katman, CNN'lerin yapı taşlarından biridir. Örnek - yukarıda bahsedilen GİRİŞ [64 × 64 × 3] üzerinde 6 filtre kullanırsak, bu [64 × 64 × 6] sesle sonuçlanabilir.
RELU−Ayrıca, önceki katmanın çıktısına bir etkinleştirme işlevi uygulayan rektifiye doğrusal birim katman olarak da adlandırılır. Başka bir şekilde, ağa RELU tarafından bir doğrusal olmayanlık eklenecektir.
POOL- Bu katman, yani Havuzlama katmanı, CNN'lerin diğer bir yapı taşıdır. Bu katmanın ana görevi aşağı örneklemedir, yani girdinin her diliminde bağımsız olarak çalışır ve onu uzamsal olarak yeniden boyutlandırır.
FC- Tam Bağlı katman veya daha özel olarak çıktı katmanı olarak adlandırılır. Çıktı sınıfı puanını hesaplamak için kullanılır ve ortaya çıkan çıktı 1 * 1 * L boyutundaki hacimdir; burada L, sınıf puanına karşılık gelen sayıdır.
Aşağıdaki diyagram CNN'lerin tipik mimarisini temsil etmektedir−
CNN yapısı oluşturma
CNN'in mimarisini ve temellerini gördük, şimdi CNTK kullanarak evrişimli ağ inşa edeceğiz. Burada önce CNN'in yapısını nasıl bir araya getireceğimizi göreceğiz ve ardından onun parametrelerini nasıl eğiteceğimize bakacağız.
Sonunda, çeşitli farklı katman kurulumlarıyla yapısını değiştirerek sinir ağını nasıl iyileştirebileceğimizi göreceğiz. MNIST resim veri setini kullanacağız.
Öyleyse önce bir CNN yapısı oluşturalım. Genel olarak, görüntülerdeki kalıpları tanımak için bir CNN oluşturduğumuzda, aşağıdakileri yaparız−
Evrişim ve havuz katmanlarının bir kombinasyonunu kullanıyoruz.
Ağın sonunda bir veya daha fazla gizli katman.
Sonunda, sınıflandırma amacıyla bir softmax katmanı ile ağı bitiriyoruz.
Aşağıdaki adımların yardımıyla ağ yapısını oluşturabiliriz−
Step 1- Öncelikle CNN için gerekli katmanları içe aktarmamız gerekiyor.
from cntk.layers import Convolution2D, Sequential, Dense, MaxPooling
Step 2- Sonra, CNN için aktivasyon fonksiyonlarını içe aktarmamız gerekiyor.
from cntk.ops import log_softmax, relu
Step 3- Bundan sonra, evrişimli katmanları daha sonra başlatmak için, glorot_uniform_initializer aşağıdaki gibi
from cntk.initializer import glorot_uniform
Step 4- Ardından, giriş değişkenleri oluşturmak için input_variableişlevi. Ve ithaldefault_option NN konfigürasyonunu biraz daha kolaylaştırmak için.
from cntk import input_variable, default_options
Step 5- Şimdi giriş görüntülerini saklamak için yeni bir input_variable. Kırmızı, yeşil ve mavi olmak üzere üç kanal içerecektir. 28 x 28 piksel boyutunda olacaktır.
features = input_variable((3,28,28))
Step 6− Sonra, başka bir tane oluşturmalıyız input_variable tahmin etmek için etiketleri saklamak.
labels = input_variable(10)
Step 7- Şimdi, default_optionNN için. Ve kullanmamız gerekiyorglorot_uniform başlatma işlevi olarak.
with default_options(initialization=glorot_uniform, activation=relu):
Step 8- Sonra, NN'nin yapısını ayarlamak için yeni bir Sequential katman kümesi.
Step 9- Şimdi bir eklememiz gerekiyor Convolutional2D ile katman filter_shape 5 ve a strides ayarı 1, içinde Sequentialkatman kümesi. Ayrıca, görüntünün orijinal boyutları korumak için doldurulması için dolguyu etkinleştirin.
model = Sequential([
Convolution2D(filter_shape=(5,5), strides=(1,1), num_filters=8, pad=True),
Step 10- Şimdi ekleme zamanı MaxPooling ile katman filter_shape 2 ve a strides görüntüyü yarıya kadar sıkıştırmak için 2 ayarı.
MaxPooling(filter_shape=(2,2), strides=(2,2)),
Step 11- Şimdi, 9. adımda yaptığımız gibi, bir tane daha eklememiz gerekiyor Convolutional2D ile katman filter_shape 5 ve a strides1 ayarı, 16 filtre kullanın. Ayrıca, dolguyu etkinleştirin, böylece önceki havuzlama katmanı tarafından üretilen görüntünün boyutu korunmalıdır.
Convolution2D(filter_shape=(5,5), strides=(1,1), num_filters=16, pad=True),
Step 12- Şimdi, 10. adımda yaptığımız gibi, bir tane daha ekleyin MaxPooling ile katman filter_shape 3 ve a strides görüntüyü üçte birine düşürmek için 3 ayarı.
MaxPooling(filter_shape=(3,3), strides=(3,3)),
Step 13- Sonunda, ağın tahmin edebileceği 10 olası sınıf için on nöronlu bir Yoğun katman ekleyin. Ağı bir sınıflandırma modeline dönüştürmek için birlog_siftmax aktivasyon işlevi.
Dense(10, activation=log_softmax)
])
CNN yapısı oluşturmak için Tam Örnek
from cntk.layers import Convolution2D, Sequential, Dense, MaxPooling
from cntk.ops import log_softmax, relu
from cntk.initializer import glorot_uniform
from cntk import input_variable, default_options
features = input_variable((3,28,28))
labels = input_variable(10)
with default_options(initialization=glorot_uniform, activation=relu):
model = Sequential([
Convolution2D(filter_shape=(5,5), strides=(1,1), num_filters=8, pad=True),
MaxPooling(filter_shape=(2,2), strides=(2,2)),
Convolution2D(filter_shape=(5,5), strides=(1,1), num_filters=16, pad=True),
MaxPooling(filter_shape=(3,3), strides=(3,3)),
Dense(10, activation=log_softmax)
])
z = model(features)
CNN'i görüntülerle eğitme
Ağın yapısını oluşturduğumuz için, ağı eğitme zamanı. Ancak ağımızın eğitimine başlamadan önce minibatch kaynakları kurmamız gerekiyor çünkü görüntülerle çalışan bir NN'yi eğitmek çoğu bilgisayarda olduğundan daha fazla bellek gerektirir.
Daha önceki bölümlerde minibatch kaynakları oluşturduk. Aşağıda, iki minibatch kaynağı kurmak için kullanılan Python kodu verilmiştir -
Bizde olduğu gibi create_datasource işlevi, modeli eğitmek için artık iki ayrı veri kaynağı (eğitim ve test etme) oluşturabiliriz.
train_datasource = create_datasource('mnist_train')
test_datasource = create_datasource('mnist_test', max_sweeps=1, train=False)
Şimdi görüntüleri hazırladığımız gibi NN'mizin eğitimine başlayabiliriz. Önceki bölümlerde yaptığımız gibi, eğitimi başlatmak için kayıp fonksiyonunda tren yöntemini kullanabiliriz. Bunun kodu aşağıdadır -
from cntk import Function
from cntk.losses import cross_entropy_with_softmax
from cntk.metrics import classification_error
from cntk.learners import sgd
@Function
def criterion_factory(output, targets):
loss = cross_entropy_with_softmax(output, targets)
metric = classification_error(output, targets)
return loss, metric
loss = criterion_factory(z, labels)
learner = sgd(z.parameters, lr=0.2)
Önceki kodun yardımıyla, NN için kaybı ve öğreniciyi ayarladık. Aşağıdaki kod, NN−'yi eğitecek ve doğrulayacaktır.
from cntk.logging import ProgressPrinter
from cntk.train import TestConfig
progress_writer = ProgressPrinter(0)
test_config = TestConfig(test_datasource)
input_map = {
features: train_datasource.streams.features,
labels: train_datasource.streams.labels
}
loss.train(train_datasource,
max_epochs=10,
minibatch_size=64,
epoch_size=60000,
parameter_learners=[learner],
model_inputs_to_streams=input_map,
callbacks=[progress_writer, test_config])
Eksiksiz Uygulama Örneği
from cntk.layers import Convolution2D, Sequential, Dense, MaxPooling
from cntk.ops import log_softmax, relu
from cntk.initializer import glorot_uniform
from cntk import input_variable, default_options
features = input_variable((3,28,28))
labels = input_variable(10)
with default_options(initialization=glorot_uniform, activation=relu):
model = Sequential([
Convolution2D(filter_shape=(5,5), strides=(1,1), num_filters=8, pad=True),
MaxPooling(filter_shape=(2,2), strides=(2,2)),
Convolution2D(filter_shape=(5,5), strides=(1,1), num_filters=16, pad=True),
MaxPooling(filter_shape=(3,3), strides=(3,3)),
Dense(10, activation=log_softmax)
])
z = model(features)
import os
from cntk.io import MinibatchSource, StreamDef, StreamDefs, ImageDeserializer, INFINITELY_REPEAT
import cntk.io.transforms as xforms
def create_datasource(folder, train=True, max_sweeps=INFINITELY_REPEAT):
mapping_file = os.path.join(folder, 'mapping.bin')
image_transforms = []
if train:
image_transforms += [
xforms.crop(crop_type='randomside', side_ratio=0.8),
xforms.scale(width=28, height=28, channels=3, interpolations='linear')
]
stream_definitions = StreamDefs(
features=StreamDef(field='image', transforms=image_transforms),
labels=StreamDef(field='label', shape=10)
)
deserializer = ImageDeserializer(mapping_file, stream_definitions)
return MinibatchSource(deserializer, max_sweeps=max_sweeps)
train_datasource = create_datasource('mnist_train')
test_datasource = create_datasource('mnist_test', max_sweeps=1, train=False)
from cntk import Function
from cntk.losses import cross_entropy_with_softmax
from cntk.metrics import classification_error
from cntk.learners import sgd
@Function
def criterion_factory(output, targets):
loss = cross_entropy_with_softmax(output, targets)
metric = classification_error(output, targets)
return loss, metric
loss = criterion_factory(z, labels)
learner = sgd(z.parameters, lr=0.2)
from cntk.logging import ProgressPrinter
from cntk.train import TestConfig
progress_writer = ProgressPrinter(0)
test_config = TestConfig(test_datasource)
input_map = {
features: train_datasource.streams.features,
labels: train_datasource.streams.labels
}
loss.train(train_datasource,
max_epochs=10,
minibatch_size=64,
epoch_size=60000,
parameter_learners=[learner],
model_inputs_to_streams=input_map,
callbacks=[progress_writer, test_config])
Çıktı
-------------------------------------------------------------------
average since average since examples
loss last metric last
------------------------------------------------------
Learning rate per minibatch: 0.2
142 142 0.922 0.922 64
1.35e+06 1.51e+07 0.896 0.883 192
[………]
Görüntü dönüşümleri
Gördüğümüz gibi, görüntü tanıma için kullanılan NN'yi eğitmek zordur ve eğitmek için çok fazla veriye ihtiyaç duyarlar. Bir sorun daha, eğitim sırasında kullanılan görüntülere fazla sığma eğiliminde olmalarıdır. Bir örnekle görelim, dik pozisyonda yüzlerin fotoğraflarını çektiğimizde, modelimiz başka bir yöne dönen yüzleri tanımakta zorlanacaktır.
Bu tür bir sorunun üstesinden gelmek için, görüntü büyütmeyi kullanabiliriz ve CNTK, görüntüler için minibatch kaynakları oluştururken belirli dönüşümleri destekler. Aşağıdaki gibi birkaç dönüşümü kullanabiliriz−
Eğitim için kullanılan görüntüleri sadece birkaç satır kodla rastgele kırpabiliriz.
Bir ölçek ve renk de kullanabiliriz.
Python kodunu takip ederek, daha önce minibatch kaynağını oluşturmak için kullanılan fonksiyonun içine bir kırpma dönüşümü ekleyerek dönüşüm listesini nasıl değiştirebileceğimizi görelim.
import os
from cntk.io import MinibatchSource, StreamDef, StreamDefs, ImageDeserializer, INFINITELY_REPEAT
import cntk.io.transforms as xforms
def create_datasource(folder, train=True, max_sweeps=INFINITELY_REPEAT):
mapping_file = os.path.join(folder, 'mapping.bin')
image_transforms = []
if train:
image_transforms += [
xforms.crop(crop_type='randomside', side_ratio=0.8),
xforms.scale(width=28, height=28, channels=3, interpolations='linear')
]
stream_definitions = StreamDefs(
features=StreamDef(field='image', transforms=image_transforms),
labels=StreamDef(field='label', shape=10)
)
deserializer = ImageDeserializer(mapping_file, stream_definitions)
return MinibatchSource(deserializer, max_sweeps=max_sweeps)
Yukarıdaki kodun yardımıyla, işlevi bir dizi görüntü dönüşümü içerecek şekilde geliştirebiliriz, böylece eğitim alırken görüntüyü rastgele kırpabiliriz, böylece görüntünün daha fazla varyasyonunu elde ederiz.