-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtrain_test_network.py
145 lines (107 loc) · 3.89 KB
/
train_test_network.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
import numpy as np
import random
import time
from progress.bar import Bar
#from simple_neural_net import Network
from matrix_neural_net import Network
import matplotlib.pyplot as plt
def target_function(x):
# multiply input by 2*np.pi to force all outputs to
# a [0,1] range
return np.sin(2 * 2 * np.pi * x)
def train_test_sin(network, num_passes=500, ratio=0.1, target_function=target_function, previous_error=None):
bar = Bar('Training', max=num_passes)
# training step
for _ in range(num_passes):
# generate a random value within the input range
x = np.random.uniform(0, 1.0)
# pass through the network (lne function only takes one input)
o = network.forward_pass([x])
# calculate the expected output value
y = target_function(x)
# backpropgate the expected value through the network
network.backward_pass(
network_input=[x],
network_output=o,
expected_output=[y])
bar.next()
bar.finish()
# testing step
error = 0.0
inputs = []
actual_outputs = []
for _ in range(int(num_passes * ratio)):
# generate random value within input range
x = np.random.uniform(0, 1.0)
# pass through the network
output = network.forward_pass([x])
# store input value for testing purposes
inputs.append(x)
# store output value for graphing purposes
actual_outputs.append(output)
expected_output = target_function(x)
# added squared difference between the expected and the actual
# to calculate loss function
error += np.sqrt((output - expected_output)**2)
# if previous_error and np.abs(error-previous_error)/previous_error < 0.1:
# network.learning_rate += 0.05
print("Error:%.4f " % error)
# return inputs and output from testing phase for graphing
return error, inputs, actual_outputs
if __name__ == "__main__":
print("instantiating network")
# TODO: Make learning rate adaptable (ie: as error goes down
# epoch to epoch, reduce learning rate, to avoid bouncing
# over global minima)
network = Network(shape=(1, 10, 25, 1))
# # set activation function of last layer to linear
# for neuron in network.output_layer:
# neuron.transfer = 'linear'
errors = []
num_epochs = 100
# prepare a graph that will update with the result of the
# testing phase after each epoch
plt.ion()
x = np.arange(0.0, 1.0, step=0.01)
expected = target_function(x)
fig, ax = plt.subplots()
reference_plot = ax.plot(x, expected, label="Expected Values")
test_plot = ax.scatter(x, expected, label="Network Output", c="r")
ax.legend()
error = None
for i in range(num_epochs):
print("iteration %i out of %i" % ((i + 1), num_epochs))
params = {
'network': network,
'num_passes': 500,
'ratio': 0.25,
'target_function': target_function
}
if error:
params['previous_error'] = error
error, test_inputs, test_outputs = train_test_sin(
**params
)
errors.append(error)
actual = []
# update data with newest test run
test_plot.set_offsets([list(i)
for i in list(zip(test_inputs, test_outputs))])
fig.canvas.draw()
fig.canvas.flush_events()
if i > 0 and i % 25 == 0:
fig.savefig("./images/epoch_%ix500.png" % i)
# save last epoch
fig.savefig("./images/epoch_%ix500.png" % i)
# stop interactive mode
plt.ioff()
# error from each epoch
print(errors)
# generate graph of loss function
fig1, ax1 = plt.subplots()
ax1.plot(errors)
ax1.set_xlabel("Training Epoch")
ax1.set_ylabel("Sum Error")
time.sleep(0.1)
fig1.savefig("./images/loss_function_500x500.png")
plt.show()