CNTK - Sinir Ağı Sınıflandırması
Bu bölümde, CNTK kullanarak sinir ağlarının nasıl sınıflandırılacağını inceleyeceğiz.
Giriş
Sınıflandırma, verilen girdi verileri için kategorisel çıktı etiketlerini veya yanıtları tahmin etme süreci olarak tanımlanabilir. Modelin eğitim aşamasında öğrendiklerine dayalı olarak kategorize edilmiş çıktı, "Siyah" veya "Beyaz" veya "istenmeyen posta" veya "istenmeyen posta yok" şeklinde olabilir.
Öte yandan, matematiksel olarak, bir eşleme işlevine yaklaşma görevidir. f girdi değişkenlerinden X, çıktı değişkenlerinden Y deyin.
Klasik bir sınıflandırma problemi örneği, e-postalarda istenmeyen posta algılama olabilir. Sadece iki çıktı kategorisi olabileceği açıktır, "spam" ve "spam yok".
Bu tür bir sınıflandırmayı uygulamak için, öncelikle "spam" ve "spam yok" e-postaların eğitim verileri olarak kullanılacağı bir sınıflandırıcı eğitimi almamız gerekir. Sınıflandırıcı başarılı bir şekilde eğitildikten sonra, bilinmeyen bir e-postayı tespit etmek için kullanılabilir.
Burada, aşağıdakilere sahip iris çiçeği veri kümesini kullanarak bir 4-5-3 NN oluşturacağız -
4-giriş düğümü (her tahmin değeri için bir tane).
5 gizli işlem düğümleri.
3 çıkışlı düğümler (çünkü iris veri kümesinde üç olası tür vardır).
Veri Kümesi Yükleniyor
İris çiçeği türlerini ayrı ayrı genişlik ve uzunluk ile taç yaprağı genişliği ve uzunluğunun fiziksel özelliklerine göre sınıflandırmak istediğimiz iris çiçeği veri kümesini kullanacağız. Veri seti, farklı iris çiçeği çeşitlerinin fiziksel özelliklerini açıklar -
Sepal uzunluk
Sepal genişlik
Petal uzunluğu
Petal genişliği
Sınıf ie iris setosa veya iris versicolor veya iris virginica
Sahibiz iris.CSVönceki bölümlerde de kullandığımız dosya. Yardımı ile yüklenebilirPandaskütüphane. Ancak sınıflandırıcımız için kullanmadan veya yüklemeden önce CNTK ile rahatlıkla kullanılabilmesi için eğitim ve test dosyalarını hazırlamamız gerekiyor.
Eğitim ve test dosyalarını hazırlama
Iris veri kümesi, makine öğrenimi projeleri için en popüler veri kümelerinden biridir. 150 veri öğesine sahiptir ve ham veriler aşağıdaki gibi görünür -
5.1 3.5 1.4 0.2 setosa
4.9 3.0 1.4 0.2 setosa
…
7.0 3.2 4.7 1.4 versicolor
6.4 3.2 4.5 1.5 versicolor
…
6.3 3.3 6.0 2.5 virginica
5.8 2.7 5.1 1.9 virginica
Daha önce de söylendiği gibi, her satırdaki ilk dört değer, farklı çeşitlerin fiziksel özelliklerini, yani Sepal uzunluğu, Sepal genişliği, Petal uzunluğu, iris çiçeklerinin Petal genişliği.
Ancak veriyi CNTK tarafından kolaylıkla kullanılabilecek formatta dönüştürmemiz gerekiyor ve bu format .ctf dosyası (önceki bölümde de bir iris.ctf oluşturduk). Aşağıdaki gibi görünecek -
|attribs 5.1 3.5 1.4 0.2|species 1 0 0
|attribs 4.9 3.0 1.4 0.2|species 1 0 0
…
|attribs 7.0 3.2 4.7 1.4|species 0 1 0
|attribs 6.4 3.2 4.5 1.5|species 0 1 0
…
|attribs 6.3 3.3 6.0 2.5|species 0 0 1
|attribs 5.8 2.7 5.1 1.9|species 0 0 1
Yukarıdaki verilerde | attribs etiketi, özellik değerinin başlangıcını işaretler ve | types, sınıf etiketi değerlerini etiketler. İstediğimiz diğer etiket isimlerini de kullanabiliriz, hatta ürün kimliği de ekleyebiliriz. Örneğin, aşağıdaki verilere bakın -
|ID 001 |attribs 5.1 3.5 1.4 0.2|species 1 0 0 |#setosa
|ID 002 |attribs 4.9 3.0 1.4 0.2|species 1 0 0 |#setosa
…
|ID 051 |attribs 7.0 3.2 4.7 1.4|species 0 1 0 |#versicolor
|ID 052 |attribs 6.4 3.2 4.5 1.5|species 0 1 0 |#versicolor
…
İris veri setinde toplam 150 veri öğesi vardır ve bu örnek için 80-20 erteleme veri kümesi kuralı kullanacağız, yani eğitim amaçlı% 80 (120 öğe) veri öğesi ve test için kalan% 20 (30 öğe) veri öğesi amaç.
Sınıflandırma modelinin oluşturulması
Öncelikle veri dosyalarını CNTK formatında işlememiz gerekiyor ve bunun için isimli helper fonksiyonunu kullanacağız. create_reader aşağıdaki gibi -
def create_reader(path, input_dim, output_dim, rnd_order, sweeps):
x_strm = C.io.StreamDef(field='attribs', shape=input_dim, is_sparse=False)
y_strm = C.io.StreamDef(field='species', shape=output_dim, is_sparse=False)
streams = C.io.StreamDefs(x_src=x_strm, y_src=y_strm)
deserial = C.io.CTFDeserializer(path, streams)
mb_src = C.io.MinibatchSource(deserial, randomize=rnd_order, max_sweeps=sweeps)
return mb_src
Şimdi, NN'miz için mimari argümanlarını ayarlamamız ve ayrıca veri dosyalarının konumunu sağlamamız gerekiyor. Aşağıdaki python kodunu kullanarak yapılabilir -
def main():
print("Using CNTK version = " + str(C.__version__) + "\n")
input_dim = 4
hidden_dim = 5
output_dim = 3
train_file = ".\\...\\" #provide the name of the training file(120 data items)
test_file = ".\\...\\" #provide the name of the test file(30 data items)
Şimdi, aşağıdaki kod satırının yardımıyla programımız eğitimsiz NN'yi oluşturacak -
X = C.ops.input_variable(input_dim, np.float32)
Y = C.ops.input_variable(output_dim, np.float32)
with C.layers.default_options(init=C.initializer.uniform(scale=0.01, seed=1)):
hLayer = C.layers.Dense(hidden_dim, activation=C.ops.tanh, name='hidLayer')(X)
oLayer = C.layers.Dense(output_dim, activation=None, name='outLayer')(hLayer)
nnet = oLayer
model = C.ops.softmax(nnet)
Şimdi, ikili eğitimsiz modeli oluşturduktan sonra, bir Öğrenci algoritması nesnesi oluşturmalı ve daha sonra bunu bir Eğitmen eğitim nesnesi oluşturmak için kullanmalıyız. SGD öğrenicisini kullanacağız vecross_entropy_with_softmax kayıp fonksiyonu -
tr_loss = C.cross_entropy_with_softmax(nnet, Y)
tr_clas = C.classification_error(nnet, Y)
max_iter = 2000
batch_size = 10
learn_rate = 0.01
learner = C.sgd(nnet.parameters, learn_rate)
trainer = C.Trainer(nnet, (tr_loss, tr_clas), [learner])
Öğrenme algoritmasını aşağıdaki gibi kodlayın -
max_iter = 2000
batch_size = 10
lr_schedule = C.learning_parameter_schedule_per_sample([(1000, 0.05), (1, 0.01)])
mom_sch = C.momentum_schedule([(100, 0.99), (0, 0.95)], batch_size)
learner = C.fsadagrad(nnet.parameters, lr=lr_schedule, momentum=mom_sch)
trainer = C.Trainer(nnet, (tr_loss, tr_clas), [learner])
Şimdi, Trainer nesnesini bitirdikten sonra, eğitim verilerini okumak için bir okuyucu işlevi oluşturmamız gerekiyor−
rdr = create_reader(train_file, input_dim, output_dim, rnd_order=True, sweeps=C.io.INFINITELY_REPEAT)
iris_input_map = { X : rdr.streams.x_src, Y : rdr.streams.y_src }
Şimdi NN modelimizi eğitmenin zamanı geldi−
for i in range(0, max_iter):
curr_batch = rdr.next_minibatch(batch_size, input_map=iris_input_map) trainer.train_minibatch(curr_batch)
if i % 500 == 0:
mcee = trainer.previous_minibatch_loss_average
macc = (1.0 - trainer.previous_minibatch_evaluation_average) * 100
print("batch %4d: mean loss = %0.4f, accuracy = %0.2f%% " \ % (i, mcee, macc))
Eğitimi tamamladıktan sonra, modeli test veri öğelerini kullanarak değerlendirelim -
print("\nEvaluating test data \n")
rdr = create_reader(test_file, input_dim, output_dim, rnd_order=False, sweeps=1)
iris_input_map = { X : rdr.streams.x_src, Y : rdr.streams.y_src }
num_test = 30
all_test = rdr.next_minibatch(num_test, input_map=iris_input_map) acc = (1.0 - trainer.test_minibatch(all_test)) * 100
print("Classification accuracy = %0.2f%%" % acc)
Eğitimli NN modelimizin doğruluğunu değerlendirdikten sonra, onu görünmeyen veriler hakkında bir tahmin yapmak için kullanacağız -
np.set_printoptions(precision = 1, suppress=True)
unknown = np.array([[6.4, 3.2, 4.5, 1.5]], dtype=np.float32)
print("\nPredicting Iris species for input features: ")
print(unknown[0]) pred_prob = model.eval(unknown)
np.set_printoptions(precision = 4, suppress=True)
print("Prediction probabilities are: ")
print(pred_prob[0])
Eksiksiz Sınıflandırma Modeli
Import numpy as np
Import cntk as C
def create_reader(path, input_dim, output_dim, rnd_order, sweeps):
x_strm = C.io.StreamDef(field='attribs', shape=input_dim, is_sparse=False)
y_strm = C.io.StreamDef(field='species', shape=output_dim, is_sparse=False)
streams = C.io.StreamDefs(x_src=x_strm, y_src=y_strm)
deserial = C.io.CTFDeserializer(path, streams)
mb_src = C.io.MinibatchSource(deserial, randomize=rnd_order, max_sweeps=sweeps)
return mb_src
def main():
print("Using CNTK version = " + str(C.__version__) + "\n")
input_dim = 4
hidden_dim = 5
output_dim = 3
train_file = ".\\...\\" #provide the name of the training file(120 data items)
test_file = ".\\...\\" #provide the name of the test file(30 data items)
X = C.ops.input_variable(input_dim, np.float32)
Y = C.ops.input_variable(output_dim, np.float32)
with C.layers.default_options(init=C.initializer.uniform(scale=0.01, seed=1)):
hLayer = C.layers.Dense(hidden_dim, activation=C.ops.tanh, name='hidLayer')(X)
oLayer = C.layers.Dense(output_dim, activation=None, name='outLayer')(hLayer)
nnet = oLayer
model = C.ops.softmax(nnet)
tr_loss = C.cross_entropy_with_softmax(nnet, Y)
tr_clas = C.classification_error(nnet, Y)
max_iter = 2000
batch_size = 10
learn_rate = 0.01
learner = C.sgd(nnet.parameters, learn_rate)
trainer = C.Trainer(nnet, (tr_loss, tr_clas), [learner])
max_iter = 2000
batch_size = 10
lr_schedule = C.learning_parameter_schedule_per_sample([(1000, 0.05), (1, 0.01)])
mom_sch = C.momentum_schedule([(100, 0.99), (0, 0.95)], batch_size)
learner = C.fsadagrad(nnet.parameters, lr=lr_schedule, momentum=mom_sch)
trainer = C.Trainer(nnet, (tr_loss, tr_clas), [learner])
rdr = create_reader(train_file, input_dim, output_dim, rnd_order=True, sweeps=C.io.INFINITELY_REPEAT)
iris_input_map = { X : rdr.streams.x_src, Y : rdr.streams.y_src }
for i in range(0, max_iter):
curr_batch = rdr.next_minibatch(batch_size, input_map=iris_input_map) trainer.train_minibatch(curr_batch)
if i % 500 == 0:
mcee = trainer.previous_minibatch_loss_average
macc = (1.0 - trainer.previous_minibatch_evaluation_average) * 100
print("batch %4d: mean loss = %0.4f, accuracy = %0.2f%% " \ % (i, mcee, macc))
print("\nEvaluating test data \n")
rdr = create_reader(test_file, input_dim, output_dim, rnd_order=False, sweeps=1)
iris_input_map = { X : rdr.streams.x_src, Y : rdr.streams.y_src }
num_test = 30
all_test = rdr.next_minibatch(num_test, input_map=iris_input_map) acc = (1.0 - trainer.test_minibatch(all_test)) * 100
print("Classification accuracy = %0.2f%%" % acc)
np.set_printoptions(precision = 1, suppress=True)
unknown = np.array([[7.0, 3.2, 4.7, 1.4]], dtype=np.float32)
print("\nPredicting species for input features: ")
print(unknown[0])
pred_prob = model.eval(unknown)
np.set_printoptions(precision = 4, suppress=True)
print("Prediction probabilities: ")
print(pred_prob[0])
if __name__== ”__main__”:
main()
Çıktı
Using CNTK version = 2.7
batch 0: mean loss = 1.0986, mean accuracy = 40.00%
batch 500: mean loss = 0.6677, mean accuracy = 80.00%
batch 1000: mean loss = 0.5332, mean accuracy = 70.00%
batch 1500: mean loss = 0.2408, mean accuracy = 100.00%
Evaluating test data
Classification accuracy = 94.58%
Predicting species for input features:
[7.0 3.2 4.7 1.4]
Prediction probabilities:
[0.0847 0.736 0.113]
Eğitimli modeli kaydetme
Bu Iris veri kümesinde yalnızca 150 veri öğesi vardır, bu nedenle NN sınıflandırıcı modelini eğitmek yalnızca birkaç saniye sürer, ancak yüz veya bin veri öğesi içeren büyük bir veri kümesi üzerinde eğitim saatler hatta günler alabilir.
Modelimizi kurtarabiliriz, böylece onu sıfırdan korumak zorunda kalmayız. Python kodunu takip etmenin yardımıyla, eğitimli NN'mizi kaydedebiliriz -
nn_classifier = “.\\neuralclassifier.model” #provide the name of the file
model.save(nn_classifier, format=C.ModelFormat.CNTKv2)
Aşağıdaki argümanlar save() yukarıda kullanılan işlev -
Dosya adı ilk argümandır save()işlevi. Dosya yolu ile birlikte de yazılabilir.
Başka bir parametre de format varsayılan değeri olan parametre C.ModelFormat.CNTKv2.
Eğitimli modeli yükleme
Eğitilen modeli kaydettikten sonra, bu modeli yüklemek çok kolaydır. Sadece kullanmamız gerekiyorload ()işlevi. Bunu aşağıdaki örnekte kontrol edelim -
import numpy as np
import cntk as C
model = C.ops.functions.Function.load(“.\\neuralclassifier.model”)
np.set_printoptions(precision = 1, suppress=True)
unknown = np.array([[7.0, 3.2, 4.7, 1.4]], dtype=np.float32)
print("\nPredicting species for input features: ")
print(unknown[0])
pred_prob = model.eval(unknown)
np.set_printoptions(precision = 4, suppress=True)
print("Prediction probabilities: ")
print(pred_prob[0])
Kaydedilen modelin yararı, kaydedilmiş bir modeli yüklediğinizde, tam olarak model henüz eğitilmiş gibi kullanılabilmesidir.