CNTK - Tekrarlayan Sinir Ağı

Şimdi CNTK'da Tekrarlayan Sinir Ağı'nın (RNN) nasıl kurulacağını anlayalım.

Giriş

Görüntüleri bir sinir ağıyla nasıl sınıflandıracağımızı öğrendik ve bu, derin öğrenmedeki ikonik işlerden biri. Ancak, sinir ağının mükemmel olduğu ve birçok araştırmanın yapıldığı bir başka alan da Tekrarlayan Sinir Ağları (RNN). Burada, RNN'nin ne olduğunu ve zaman serisi verileriyle uğraşmamız gereken senaryolarda nasıl kullanılabileceğini bileceğiz.

Tekrarlayan Sinir Ağı nedir?

Tekrarlayan sinir ağları (RNN'ler), zaman içinde muhakeme yapabilen özel NN türü olarak tanımlanabilir. RNN'ler esas olarak, zamanla değişen değerlerle, yani zaman serisi verileriyle ilgilenmemiz gereken senaryolarda kullanılır. Bunu daha iyi anlamak için, normal sinir ağları ile tekrarlayan sinir ağları arasında küçük bir karşılaştırma yapalım -

  • Bildiğimiz gibi, normal bir sinir ağında sadece bir girdi sağlayabiliriz. Bu, yalnızca bir tahminle sonuçlanmasını sınırlar. Size bir örnek vermek gerekirse, normal sinir ağlarını kullanarak metin çevirme işi yapabiliriz.

  • Öte yandan, tekrarlayan sinir ağlarında, tek bir tahminle sonuçlanan bir dizi örnek sağlayabiliriz. Başka bir deyişle, RNN'leri kullanarak, bir girdi dizisine dayalı bir çıktı dizisini tahmin edebiliriz. Örneğin, çeviri görevlerinde RNN ile epeyce başarılı deneyler yapılmıştır.

Tekrarlayan Sinir Ağının Kullanımları

RNN'ler birkaç şekilde kullanılabilir. Bunlardan bazıları aşağıdaki gibidir -

Tek bir çıktıyı tahmin etmek

RNN'nin bir diziye göre tek bir çıktıyı nasıl tahmin edebileceğini adımlara derinlemesine incelemeden önce, temel bir RNN'nin nasıl göründüğüne bakalım

Yukarıdaki diyagramda yapabileceğimiz gibi, RNN girişe bir geri döngü bağlantısı içerir ve ne zaman, bir dizi değer beslediğimizde, sıradaki her bir öğeyi zaman adımları olarak işleyecektir.

Ayrıca, geridöngü bağlantısı nedeniyle, RNN üretilen çıktıyı dizideki bir sonraki eleman için girdiyle birleştirebilir. Bu şekilde, RNN bir tahmin yapmak için kullanılabilecek tüm dizi üzerinde bir bellek oluşturacaktır.

RNN ile tahmin yapmak için aşağıdaki adımları uygulayabiliriz

  • İlk olarak, bir başlangıç ​​gizli durumu oluşturmak için, giriş dizisinin ilk öğesini beslememiz gerekir.

  • Bundan sonra, güncellenmiş bir gizli durum oluşturmak için, ilk gizli durumu alıp giriş sırasındaki ikinci öğe ile birleştirmemiz gerekir.

  • Sonunda, son gizli durumu üretmek ve RNN için çıktıyı tahmin etmek için, girdi dizisindeki son öğeyi almamız gerekir.

Bu şekilde, bu geridöngü bağlantısının yardımıyla bir RNN'ye zaman içinde meydana gelen kalıpları tanımasını öğretebiliriz.

Bir sekansı tahmin etmek

Yukarıda tartışılan temel RNN modeli, diğer kullanım durumlarına da genişletilebilir. Örneğin, tek bir girdiye dayalı olarak bir dizi değeri tahmin etmek için kullanabiliriz. Bu senaryoda, RNN ile tahmin yapmak için aşağıdaki adımları gerçekleştirebiliriz -

  • İlk olarak, bir ilk gizli durum oluşturmak ve çıktı dizisindeki ilk öğeyi tahmin etmek için, sinir ağına bir girdi örneği beslememiz gerekir.

  • Bundan sonra, güncellenmiş bir gizli durum ve çıktı dizisindeki ikinci öğeyi üretmek için, ilk gizli durumu aynı örnekle birleştirmemiz gerekir.

  • Sonunda, gizli durumu bir kez daha güncellemek ve çıktı dizisindeki son öğeyi tahmin etmek için örneği başka bir zaman besleriz.

Dizileri tahmin etme

Bir diziye dayalı olarak tek bir değeri nasıl tahmin edeceğimizi ve tek bir değere dayalı bir diziyi nasıl tahmin edeceğimizi gördüğümüz gibi. Şimdi diziler için dizileri nasıl tahmin edebileceğimize bakalım. Bu senaryoda, RNN ile tahmin yapmak için aşağıdaki adımları gerçekleştirebiliriz -

  • İlk olarak, bir ilk gizli durum oluşturmak ve çıktı dizisindeki ilk öğeyi tahmin etmek için, girdi dizisindeki ilk öğeyi almamız gerekir.

  • Bundan sonra, gizli durumu güncellemek ve çıktı dizisindeki ikinci öğeyi tahmin etmek için ilk gizli durumu almamız gerekir.

  • Sonunda, çıktı dizisindeki son öğeyi tahmin etmek için, güncellenmiş gizli durumu ve giriş dizisindeki son öğeyi almamız gerekir.

RNN'nin Çalışması

Tekrarlayan sinir ağlarının (RNN'ler) çalışmasını anlamak için önce ağdaki tekrarlayan katmanların nasıl çalıştığını anlamamız gerekir. Öyleyse önce, e'nin çıktıyı standart bir tekrarlayan katmanla nasıl tahmin edebileceğini tartışalım.

Standart RNN katmanıyla çıktı tahmin etme

Daha önce tartıştığımız gibi, RNN'deki temel bir katmanın, bir sinir ağındaki normal bir katmandan oldukça farklı olduğu. Önceki bölümde, diyagramda ayrıca RNN'nin temel mimarisini gösterdik. Gizli durumu ilk kez adım adım sırayla güncellemek için aşağıdaki formülü kullanabiliriz -

Yukarıdaki denklemde, başlangıçtaki gizli durum ile bir dizi ağırlık arasındaki iç çarpımı hesaplayarak yeni gizli durumu hesaplıyoruz.

Şimdi bir sonraki adım için, mevcut zaman adımının gizli durumu, dizideki bir sonraki adım için ilk gizli durum olarak kullanılır. Bu nedenle, gizli durumu ikinci sefer adımı için güncellemek için, ilk adımda yapılan hesaplamaları aşağıdaki gibi tekrar edebiliriz -

Ardından, sırayla üçüncü ve son adım için gizli durumu güncelleme işlemini aşağıdaki gibi tekrarlayabiliriz -

Ve sırayla yukarıdaki tüm adımları işlediğimizde, çıktıyı şu şekilde hesaplayabiliriz -

Yukarıdaki formül için, üçüncü bir ağırlık seti ve son zaman adımından itibaren gizli durumu kullandık.

Gelişmiş Tekrarlayan Birimler

Temel tekrarlayan katmanla ilgili ana sorun, kaybolan gradyan problemidir ve bu nedenle uzun vadeli korelasyonları öğrenmede çok iyi değildir. Basit bir deyişle, temel yinelenen katman uzun dizileri çok iyi işlemez. Daha uzun dizilerle çalışmak için çok daha uygun olan diğer bazı tekrarlayan katman türlerinin aşağıdaki gibi olmasının nedeni budur -

Uzun-Kısa Süreli Bellek (LSTM)

Uzun-kısa süreli bellek (LSTM'ler) ağları Hochreiter & Schmidhuber tarafından tanıtıldı. Bir şeyleri uzun süre hatırlamak için temel bir yinelenen katman elde etme sorununu çözdü. LSTM'nin mimarisi yukarıdaki şemada verilmiştir. Gördüğümüz gibi, giriş nöronlarına, hafıza hücrelerine ve çıkış nöronlarına sahiptir. Kaybolan gradyan sorunuyla mücadele etmek için, Uzun-kısa süreli bellek ağları açık bir bellek hücresi (önceki değerleri depolar) ve aşağıdaki kapıları kullanır -

  • Forget gate- Adından da anlaşılacağı gibi, hafıza hücresine önceki değerleri unutmasını söyler. Bellek hücresi değerleri, geçit yani 'geçidi unut' ona onları unutmasını söyleyene kadar saklar.

  • Input gate- Adından da anlaşılacağı gibi, hücreye yeni şeyler ekler.

  • Output gate- Adından da anlaşılacağı gibi, çıkış kapısı vektörlerin hücreden sonraki gizli duruma ne zaman geçeceğine karar verir.

Geçitli Tekrarlayan Birimler (GRU'lar)

Gradient recurrent units(GRU'lar), LSTM ağının küçük bir varyasyonudur. Bir tane daha az geçidi vardır ve LSTM'lerden biraz farklı kablolanmıştır. Mimarisi yukarıdaki diyagramda gösterilmektedir. Giriş nöronlarına, geçitli bellek hücrelerine ve çıkış nöronlarına sahiptir. Geçitli Tekrarlayan Birimler ağı aşağıdaki iki kapıya sahiptir -

  • Update gate- Aşağıdaki iki şeyi belirler−

    • Son durumdan ne kadar bilgi saklanmalıdır?

    • Önceki katmandan ne kadar bilgi alınmalıdır?

  • Reset gate- Kapıyı sıfırlamanın işlevselliği, LSTM ağının unutma geçidine çok benzer. Tek fark, biraz farklı yerleştirilmiş olmasıdır.

Uzun-kısa süreli bellek ağının aksine, Geçitli Tekrarlayan Ünite ağları biraz daha hızlıdır ve çalıştırılması daha kolaydır.

RNN yapısı oluşturma

Başlamadan önce, veri kaynağımızın herhangi birinden gelen çıktı hakkında öngörüde bulunmak için önce RNN'yi oluşturmalıyız ve RNN'yi inşa etmek, önceki bölümde normal sinir ağı oluşturduğumuzla tamamen aynı. Bir tane oluşturmak için kod aşağıdadır−

from cntk.losses import squared_error
from cntk.io import CTFDeserializer, MinibatchSource, INFINITELY_REPEAT, StreamDefs, StreamDef
from cntk.learners import adam
from cntk.logging import ProgressPrinter
from cntk.train import TestConfig
BATCH_SIZE = 14 * 10
EPOCH_SIZE = 12434
EPOCHS = 10

Birden çok katmanı üst üste koymak

CNTK'da birden fazla tekrarlayan katmanı da istifleyebiliriz. Örneğin, aşağıdaki katman kombinasyonunu kullanabiliriz−

from cntk import sequence, default_options, input_variable
from cntk.layers import Recurrence, LSTM, Dropout, Dense, Sequential, Fold
features = sequence.input_variable(1)
with default_options(initial_state = 0.1):
   model = Sequential([
      Fold(LSTM(15)),
      Dense(1)
   ])(features)
target = input_variable(1, dynamic_axes=model.dynamic_axes)

Yukarıdaki kodda görebileceğimiz gibi, CNTK'da RNN'yi modelleyebilmemiz için aşağıdaki iki yola sahibiz:

  • İlk olarak, yinelenen bir katmanın yalnızca son çıktısını istiyorsak, Fold GRU, LSTM veya hatta RNNStep gibi tekrarlayan bir katmanla kombinasyon halinde katman.

  • İkinci olarak, alternatif bir yol olarak, Recurrence blok.

Zaman serisi verileriyle RNN eğitimi

Modeli oluşturduktan sonra, CNTK'da RNN'yi nasıl eğitebileceğimize bakalım -

from cntk import Function
@Function
def criterion_factory(z, t):
   loss = squared_error(z, t)
   metric = squared_error(z, t)
   return loss, metric
loss = criterion_factory(model, target)
learner = adam(model.parameters, lr=0.005, momentum=0.9)

Şimdi verileri eğitim sürecine yüklemek için, bir dizi CTF dosyasından dizileri seriyi kaldırmamız gerekir. Aşağıdaki kod,create_datasource hem eğitim hem de test veri kaynağını oluşturmak için yararlı bir yardımcı program işlevi olan işlev.

target_stream = StreamDef(field='target', shape=1, is_sparse=False)
features_stream = StreamDef(field='features', shape=1, is_sparse=False)
deserializer = CTFDeserializer(filename, StreamDefs(features=features_stream, target=target_stream))
   datasource = MinibatchSource(deserializer, randomize=True, max_sweeps=sweeps)
return datasource
train_datasource = create_datasource('Training data filename.ctf')#we need to provide the location of training file we created from our dataset.
test_datasource = create_datasource('Test filename.ctf', sweeps=1) #we need to provide the location of testing file we created from our dataset.

Artık veri kaynaklarını, modelini ve kayıp fonksiyonunu kurduğumuz için eğitim sürecini başlatabiliriz. Temel sinir ağları ile önceki bölümlerde yaptığımız gibi oldukça benzer.

progress_writer = ProgressPrinter(0)
test_config = TestConfig(test_datasource)
input_map = {
   features: train_datasource.streams.features,
   target: train_datasource.streams.target
}
history = loss.train(
   train_datasource,
   epoch_size=EPOCH_SIZE,
   parameter_learners=[learner],
   model_inputs_to_streams=input_map,
   callbacks=[progress_writer, test_config],
   minibatch_size=BATCH_SIZE,
   max_epochs=EPOCHS
)

Çıktıyı aşağıdaki gibi alacağız -

Çıktı−

average  since  average  since  examples
loss      last  metric  last
------------------------------------------------------
Learning rate per minibatch: 0.005
0.4      0.4    0.4      0.4      19
0.4      0.4    0.4      0.4      59
0.452    0.495  0.452    0.495   129
[…]

Modeli doğrulama

Aslında bir RNN ile yeniden biçimlendirme, diğer herhangi bir CNK modeliyle tahmin yapmaya oldukça benzer. Tek fark, tek örnekler yerine diziler sağlamamız gerektiğidir.

Şimdi, RNN'miz nihayet eğitimle tamamlandığı için, modeli aşağıdaki gibi birkaç örnek dizisi kullanarak test ederek doğrulayabiliriz -

import pickle
with open('test_samples.pkl', 'rb') as test_file:
test_samples = pickle.load(test_file)
model(test_samples) * NORMALIZE

Çıktı−

array([[ 8081.7905],
[16597.693 ],
[13335.17 ],
...,
[11275.804 ],
[15621.697 ],
[16875.555 ]], dtype=float32)