-
Notifications
You must be signed in to change notification settings - Fork 1
Expand file tree
/
Copy pathNeuralNetwork.mjs
More file actions
89 lines (76 loc) · 3.13 KB
/
NeuralNetwork.mjs
File metadata and controls
89 lines (76 loc) · 3.13 KB
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
import { Neuron, InputNeuron, OutputNeuron } from "./Neuron";
import { NeuralLayer } from "./NeuralLayer";
import { Answer } from './Answer';
import { Sigmoid } from "./Activators";
import Promise from 'bluebird';
export class NeuralNetwork {
constructor(inputCount, hiddenCount, outputCount, learningRate = 0.01, neuronActivator = Sigmoid) {
this.inputLayer = new NeuralLayer(inputCount, InputNeuron, neuronActivator);
this.hiddenLayer = new NeuralLayer(hiddenCount, Neuron, neuronActivator);
this.outputLayer = new NeuralLayer(outputCount, OutputNeuron, neuronActivator);
this.learningRate = learningRate;
this.inputLayer.connectToAll(this.hiddenLayer);
this.hiddenLayer.connectToAll(this.outputLayer);
}
pulse() {
let feedforwardLayer = this.inputLayer.nextLayer;
while (feedforwardLayer) {
feedforwardLayer.pulse();
feedforwardLayer = feedforwardLayer.nextLayer;
}
}
backpropagation(desiredResults) {
let backpropagatingLayer = this.outputLayer;
while (backpropagatingLayer !== this.inputLayer) {
backpropagatingLayer.backpropagate(desiredResults);
backpropagatingLayer = backpropagatingLayer.prevLayer;
}
}
async train(inputArray, expectedOutput, trainCount = 5000) {
const allSteps = trainCount * inputArray.length;
for (let trainStep = 0; trainStep < trainCount; trainStep++) {
for (let inputIndex = 0; inputIndex < inputArray.length; inputIndex++) {
await this.trainStep(inputArray[inputIndex], expectedOutput[inputIndex]);
let currentStep = trainStep * inputArray.length + inputIndex;
process.stdout.cursorTo(0);
process.stdout.write(`${(Math.round(((currentStep) / allSteps) * 10000) / 100) }%`);
}
}
}
trainStep(input, desiredResults) {
return new Promise((resolve) => {
this.ask(input);
this.backpropagation(desiredResults);
let feedforwardLayer = this.inputLayer.nextLayer;
while (feedforwardLayer) {
feedforwardLayer.train(this.learningRate);
feedforwardLayer = feedforwardLayer.nextLayer;
}
setImmediate(resolve);
});
}
ask(input) {
this.inputLayer.getNeurons().forEach((neuron, index) => neuron.output = input[index]);
this.pulse();
return new Answer([...this.outputLayer.neurons.map(neuron => neuron.output)]);
}
exportNeuronsAsJson() {
const nodes = [];
this.inputLayer.getNeurons().forEach((neuron, index) => nodes.push({
label: `input${index}`,
layer: 1,
neuron
}));
this.hiddenLayer.getNeurons().forEach((neuron, index) => nodes.push({
label: `hidden${index}`,
layer: 2,
neuron
}));
this.outputLayer.getNeurons().forEach((neuron, index) => nodes.push({
label: `output${index}`,
layer: 3,
neuron
}));
return JSON.stringify({ nodes });
}
}