CNTK-신경망 회귀

이 장은 CNTK와 관련된 신경망 회귀를 이해하는 데 도움이 될 것입니다.

소개

알고 있듯이 하나 이상의 예측 변수에서 숫자 값을 예측하기 위해 회귀를 사용합니다. 예를 들어 100 개 도시 중 하나에서 집의 중앙값을 예측하는 예를 들어 보겠습니다. 이를 위해 다음을 포함하는 데이터가 있습니다.

  • 각 마을에 대한 범죄 통계.

  • 각 마을의 집의 나이.

  • 각 도시에서 주요 위치까지의 거리 측정.

  • 각 도시의 학생 대 교사 비율.

  • 각 도시에 대한 인종 인구 통계 통계입니다.

  • 각 마을의 평균 주택 가치.

이 다섯 가지 예측 변수를 기반으로 주택 중앙값을 예측하려고합니다. 이를 위해 우리는 다음과 같은 선을 따라 선형 회귀 모델을 만들 수 있습니다.

Y = a0+a1(crime)+a2(house-age)+(a3)(distance)+(a4)(ratio)+(a5)(racial)

위의 방정식에서-

Y 예측 된 중앙값입니다.

a0은 상수이고

a1에서 a5는 모두 위에서 논의한 5 개의 예측 변수와 관련된 상수입니다.

또한 신경망을 사용하는 다른 접근 방식도 있습니다. 보다 정확한 예측 모델을 생성합니다.

여기에서는 CNTK를 사용하여 신경망 회귀 모델을 생성합니다.

데이터 세트로드

CNTK를 사용하여 신경망 회귀를 구현하기 위해 Boston 지역 주택 값 데이터 세트를 사용합니다. 데이터 세트는 UCI Machine Learning Repository에서 다운로드 할 수 있습니다.https://archive.ics.uci.edu/ml/machine-learning-databases/housing/. 이 데이터 세트에는 총 14 개의 변수와 506 개의 인스턴스가 있습니다.

그러나 구현 프로그램에서는 14 개의 변수 중 6 개와 100 개의 인스턴스를 사용할 것입니다. 6 개 중 5 개는 예측 변수로, 하나는 예측할 값으로 사용됩니다. 100 개의 인스턴스에서 학습용으로 80 개를 사용하고 테스트 용으로 20 개를 사용합니다. 우리가 예측하고 싶은 가치는 마을의 평균 주택 가격입니다. 우리가 사용할 5 가지 예측 변수를 봅시다.

  • Crime per capita in the town −이 예측 변수와 관련된 더 작은 값을 기대합니다.

  • Proportion of owner − 1940 년 이전에 지어진 점유 단위-값이 클수록 오래된 집을 의미하므로이 예측 변수와 관련된 더 작은 값이 예상됩니다.

  • Weighed distance of the town to five Boston employment centers.

  • Area school pupil-to-teacher ratio.

  • An indirect metric of the proportion of black residents in the town.

교육 및 테스트 파일 준비

이전과 마찬가지로 먼저 원시 데이터를 CNTK 형식으로 변환해야합니다. 훈련 목적으로 처음 80 개의 데이터 항목을 사용할 것이므로 탭으로 구분 된 CNTK 형식은 다음과 같습니다.

|predictors 1.612820 96.90 3.76 21.00 248.31 |medval 13.50
|predictors 0.064170 68.20 3.36 19.20 396.90 |medval 18.90
|predictors 0.097440 61.40 3.38 19.20 377.56 |medval 20.00
. . .

다음 20 개 항목도 CNTK 형식으로 변환되어 테스트 목적으로 사용됩니다.

회귀 모델 구성

먼저 데이터 파일을 CNTK 형식으로 처리해야하며이를 위해 이름이 지정된 도우미 함수를 사용합니다. create_reader 다음과 같이-

def create_reader(path, input_dim, output_dim, rnd_order, sweeps):
x_strm = C.io.StreamDef(field='predictors', shape=input_dim, is_sparse=False)
y_strm = C.io.StreamDef(field='medval', 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

다음으로 CNTK 미니 배치 개체를 받아들이고 사용자 지정 정확도 메트릭을 계산하는 도우미 함수를 만들어야합니다.

def mb_accuracy(mb, x_var, y_var, model, delta):
   num_correct = 0
   num_wrong = 0
   x_mat = mb[x_var].asarray()
   y_mat = mb[y_var].asarray()
for i in range(mb[x_var].shape[0]):
  v = model.eval(x_mat[i])
  y = y_mat[i]
if np.abs(v[0,0] – y[0,0]) < delta:
   num_correct += 1
else:
   num_wrong += 1
return (num_correct * 100.0)/(num_correct + num_wrong)

이제 NN에 대한 아키텍처 인수를 설정하고 데이터 파일의 위치도 제공해야합니다. 다음 파이썬 코드의 도움으로 할 수 있습니다-

def main():
print("Using CNTK version = " + str(C.__version__) + "\n")
input_dim = 5
hidden_dim = 20
output_dim = 1
train_file = ".\\...\\" #provide the name of the training file(80 data items)
test_file = ".\\...\\" #provide the name of the test file(20 data items)

이제 다음 코드 라인의 도움으로 우리 프로그램은 훈련되지 않은 NN을 생성합니다.

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)
model = C.ops.alias(oLayer)

이제 훈련되지 않은 이중 모델을 만들었 으면 학습자 알고리즘 개체를 설정해야합니다. SGD 학습자를 사용하고squared_error 손실 함수-

tr_loss = C.squared_error(model, Y)
max_iter = 3000
batch_size = 5
base_learn_rate = 0.02
sch=C.learning_parameter_schedule([base_learn_rate, base_learn_rate/2], minibatch_size=batch_size, epoch_size=int((max_iter*batch_size)/2))
learner = C.sgd(model.parameters, sch)
trainer = C.Trainer(model, (tr_loss), [learner])

이제 학습 알고리즘 객체를 완료하면 학습 데이터를 읽을 리더 함수를 만들어야합니다.

rdr = create_reader(train_file, input_dim, output_dim, rnd_order=True, sweeps=C.io.INFINITELY_REPEAT)
boston_input_map = { X : rdr.streams.x_src, Y : rdr.streams.y_src }

이제 NN 모델을 훈련 할 시간입니다.

for i in range(0, max_iter):
curr_batch = rdr.next_minibatch(batch_size, input_map=boston_input_map) trainer.train_minibatch(curr_batch)
if i % int(max_iter/10) == 0:
mcee = trainer.previous_minibatch_loss_average
acc = mb_accuracy(curr_batch, X, Y, model, delta=3.00)
print("batch %4d: mean squared error = %8.4f, accuracy = %5.2f%% " \ % (i, mcee, acc))

학습이 끝나면 테스트 데이터 항목을 사용하여 모델을 평가 해 보겠습니다.

print("\nEvaluating test data \n")
rdr = create_reader(test_file, input_dim, output_dim, rnd_order=False, sweeps=1)
boston_input_map = { X : rdr.streams.x_src, Y : rdr.streams.y_src }
num_test = 20
all_test = rdr.next_minibatch(num_test, input_map=boston_input_map)
acc = mb_accuracy(all_test, X, Y, model, delta=3.00)
print("Prediction accuracy = %0.2f%%" % acc)

훈련 된 NN 모델의 정확성을 평가 한 후, 우리는 보이지 않는 데이터에 대한 예측을 위해이를 사용할 것입니다.

np.set_printoptions(precision = 2, suppress=True)
unknown = np.array([[0.09, 50.00, 4.5, 17.00, 350.00], dtype=np.float32)
print("\nPredicting median home value for feature/predictor values: ")
print(unknown[0])
pred_prob = model.eval({X: unknown)
print("\nPredicted value is: ")
print(“$%0.2f (x1000)” %pred_value[0,0])

완전한 회귀 모델

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='predictors', shape=input_dim, is_sparse=False)
y_strm = C.io.StreamDef(field='medval', 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 mb_accuracy(mb, x_var, y_var, model, delta):
num_correct = 0
num_wrong = 0
x_mat = mb[x_var].asarray()
y_mat = mb[y_var].asarray()
for i in range(mb[x_var].shape[0]):
   v = model.eval(x_mat[i])
   y = y_mat[i]
if np.abs(v[0,0] – y[0,0]) < delta:
   num_correct += 1
else:
   num_wrong += 1
return (num_correct * 100.0)/(num_correct + num_wrong)
def main():
print("Using CNTK version = " + str(C.__version__) + "\n")
input_dim = 5
hidden_dim = 20
output_dim = 1
train_file = ".\\...\\" #provide the name of the training file(80 data items)
test_file = ".\\...\\" #provide the name of the test file(20 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)
model = C.ops.alias(oLayer)
tr_loss = C.squared_error(model, Y)
max_iter = 3000
batch_size = 5
base_learn_rate = 0.02
sch = C.learning_parameter_schedule([base_learn_rate, base_learn_rate/2], minibatch_size=batch_size, epoch_size=int((max_iter*batch_size)/2))
learner = C.sgd(model.parameters, sch)
trainer = C.Trainer(model, (tr_loss), [learner])
rdr = create_reader(train_file, input_dim, output_dim, rnd_order=True, sweeps=C.io.INFINITELY_REPEAT)
boston_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=boston_input_map) trainer.train_minibatch(curr_batch)
if i % int(max_iter/10) == 0:
   mcee = trainer.previous_minibatch_loss_average
   acc = mb_accuracy(curr_batch, X, Y, model, delta=3.00)
   print("batch %4d: mean squared error = %8.4f, accuracy = %5.2f%% " \ % (i, mcee, acc))
   print("\nEvaluating test data \n")
   rdr = create_reader(test_file, input_dim, output_dim, rnd_order=False, sweeps=1)
   boston_input_map = { X : rdr.streams.x_src, Y : rdr.streams.y_src }
   num_test = 20
all_test = rdr.next_minibatch(num_test, input_map=boston_input_map)
acc = mb_accuracy(all_test, X, Y, model, delta=3.00)
print("Prediction accuracy = %0.2f%%" % acc)
np.set_printoptions(precision = 2, suppress=True)
unknown = np.array([[0.09, 50.00, 4.5, 17.00, 350.00], dtype=np.float32)
print("\nPredicting median home value for feature/predictor values: ")
print(unknown[0])
pred_prob = model.eval({X: unknown)
print("\nPredicted value is: ")
print(“$%0.2f (x1000)” %pred_value[0,0])
if __name__== ”__main__”:
   main()

산출

Using CNTK version = 2.7
batch 0: mean squared error = 385.6727, accuracy = 0.00%
batch 300: mean squared error = 41.6229, accuracy = 20.00%
batch 600: mean squared error = 28.7667, accuracy = 40.00%
batch 900: mean squared error = 48.6435, accuracy = 40.00%
batch 1200: mean squared error = 77.9562, accuracy = 80.00%
batch 1500: mean squared error = 7.8342, accuracy = 60.00%
batch 1800: mean squared error = 47.7062, accuracy = 60.00%
batch 2100: mean squared error = 40.5068, accuracy = 40.00%
batch 2400: mean squared error = 46.5023, accuracy = 40.00%
batch 2700: mean squared error = 15.6235, accuracy = 60.00%
Evaluating test data
Prediction accuracy = 64.00%
Predicting median home value for feature/predictor values:
[0.09 50. 4.5 17. 350.]
Predicted value is:
$21.02(x1000)

훈련 된 모델 저장

이 Boston Home 값 데이터 세트에는 506 개의 데이터 항목 만 있습니다 (이 중 100 개만 소송을 제기 함). 따라서 NN 회귀 모델을 훈련하는 데 몇 초 밖에 걸리지 않지만 수백 또는 수천 개의 데이터 항목이있는 대규모 데이터 세트에 대한 훈련은 몇 시간 또는 며칠이 걸릴 수 있습니다.

모델을 저장할 수 있으므로 처음부터 유지할 필요가 없습니다. 다음 Python 코드의 도움으로 훈련 된 NN을 저장할 수 있습니다.

nn_regressor = “.\\neuralregressor.model” #provide the name of the file
model.save(nn_regressor, format=C.ModelFormat.CNTKv2)

다음은 위에서 사용 된 save () 함수의 인수입니다.

  • 파일 이름은 다음의 첫 번째 인수입니다. save()함수. 파일 경로와 함께 쓸 수도 있습니다.

  • 또 다른 매개 변수는 format 기본값이있는 매개 변수 C.ModelFormat.CNTKv2.

훈련 된 모델로드

훈련 된 모델을 저장 한 후에는 해당 모델을로드하기가 매우 쉽습니다. load () 함수 만 사용하면됩니다. 다음 예제에서 이것을 확인하십시오-

import numpy as np
import cntk as C
model = C.ops.functions.Function.load(“.\\neuralregressor.model”)
np.set_printoptions(precision = 2, suppress=True)
unknown = np.array([[0.09, 50.00, 4.5, 17.00, 350.00], dtype=np.float32)
print("\nPredicting area median home value for feature/predictor values: ")
print(unknown[0])
pred_prob = model.eval({X: unknown)
print("\nPredicted value is: ")
print(“$%0.2f (x1000)” %pred_value[0,0])

저장된 모델의 장점은 저장된 모델을로드하면 모델이 방금 학습 된 것처럼 정확히 사용할 수 있다는 것입니다.