Übung 4.4 zu Einführung in Maschinelles Lernen (PD Stefan Bosse) |
Daten können als Elemente der this
Variable gespeichert und wieder geladen werden (wie z.B. bereits verarbeitete Daten, Modelle, usw.)
Wenn in einer Funktion (im Körper) auf das Notebook this
zugegriffen werden soll, muss außerhalb der Funktion eine Referenz zu einer lokalen Variable var self=this;
hergestellt werden. Z. B.
this.data=[];
var self=this;
function foo( ) {
self.push([1,2,3,4])
}
DATA: Variable State.dataEE01 Type: {X1, X2, X3, X4, X5, X6, X7, X8, Y1, Y2} []
X1 Relative Compactness
X2 Surface Area
X3 Wall Area
X4 Roof Area
X5 Overall Height
X6 Orientation
X7 Glazing Area
X8 Glazing Area Distribution
y1 Heating Load
y2 Cooling Load
X,Y=Klasse={N,P} 1
2
print(dataEE01);
Table(dataEE01);
▸
|
✗
≡
|
this
(eine Datenstruktur) ausgetauscht. D.h. wenn z.B. in einem Snippet Daten erzeugt werden dann dort this.data=[1,2,3]
, und in einem anderen Snippet können diese dann mit this.data
wieder abgerufen (und natürlich auch verändert) werden.Nachfolgend wird gezeigt wie auf einfachen numerischen oder kategorischen Daten eine einfache Prädiktorfunktion algorithmisch durch Tarining erzeugt werden kann.
ML
verwendet.Es gibt für alle Algorithmen und Verfahren einen gemeinsamen Satz an Operationen:
ML.learner
: Eine neue Trainingsinatanz erstellen ML.train
: Mit Trainingsdaten trainierenML.predict
: Inferenz eines trainiertesnModellsML.test
: Inferenz eines trainiertesn Modells mit Testdaten und Ausgabe von Fehlerstatistiken (momentan unvollstänig; daher mit ML.predict
arbeiten und eigene Fehleranalyse durchführen)ML.print
: Ausgabe der Modellstruktur (wird nicht für alle Algorithmen unterstützt)Anders als bei Entscheidungsbaumlernern wird hier zwischen Modellerzeugung und Modellanpassung (Training) unterschieden
Neuronale Netze können nur numerische Variablen verarbeiten (sowohl Eingabeattribute als auch Zielvariablen)
Eine Datennormalisierung ist ratsam, d.h. alle Ein- und Ausgabevariablen sollen einen Wertebereich von [0,1] abdecken
Kategorische Variablen müssen in numerische Werte kodiert werden;
Zunächst werden bereits normalisierte Daten angenommen werden.
Mittels der ML.split
Funktion können Datentabellen randomisiert in zwei Partitionen aufgeteilt werden (Training und Test)
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
var x = [[1,1,1,0,0,0],
[1,0,1,0,0,0],
[1,1,1,0,0,0],
[0,0,1,1,1,0],
[0,0,1,1,0,0],
[1,0,1,0,0,0],
[1,1,1,0,0,0],
[0,0,1,1,1,0],
[0,0,1,1,0,0],
[0,0,1,1,1,0]];
var y = [[1, 0],
[1, 0],
[1, 0],
[0, 1],
[0, 1],
[1, 0],
[1, 0],
[0, 1],
[0, 1],
[0, 1]];
this.data0 = x.merge(y,'c');
var parts = ML.split(this.data0,5,5); // N ist hier zu klein; daher ist Aufteilung nicht sinnvoll!
// this.data0Train=parts[0]; this.data0Test=parts[0];
this.data0Train=this.data0; this.data0Test=this.data0;
Table(this.data0);
▸
|
✗
≡
|
Nachfolgend eine beispielhafte Implementierung der Modellfunktion M(x): x → y durch den MLP Algorithmus (MLP: Multilayer Perceptron)
Der MLP Algorithmus basiert auf einem vereinfachten SMO Trainingsverfahren
Die Konfiguration des MLP Netzwerkes erfordert die Angabe der Eingangsneuronen (hier 6), der inenren Schichten (wenn überhaupt), und der Anzahl der Ausgangsneuronen (hier 2)
1
2
3
4
5
6
// Erzeugung des Modells
this.model0 = ML.learner({
algorithm:ML.ML.MLP,
layers: [6,4,4,5,2],
verbose : 1,
});
▸
|
✗
≡
|
Beispiel Training Prädiktorfunktion MLPAnders als bei Entscheidungsbäumen können KNN mehrere Zielvariablen ausgeben!
1
2
3
4
5
6
7
8
// Erzeugung des Modells
ML.log(print);
ML.train(this.model0,{
data : this.data0,
features : [0,1,2,3,4,5],
target : [6,7],
epochs : 100,
});
▸
|
✗
≡
|
1
2
3
4
5
6
7
8
9
10
11
var self=this;
var dataX0 = this.data0.sub([],[0,5]);
var dataY0 = this.data0.sub([],[6,7]);
var results = ML.predict(this.model0,dataX0);
var error=0; // quadr. Fehler
results.forEach(function (row,index) {
error += Math.pow(row[0]-dataY0[index][0],2);
});
results=results.merge(dataY0,'c');
Table(results);
print(error)
▸
|
✗
≡
|
1
2
3
4
5
var self=this;
var dataX0 = this.data0.sub([],[0,5]);
var dataY0 = this.data0.sub([],[6,7]);
var results = ML.predict(this.model0,dataX0);
print(ML.statistics.crossEntropy(results,dataY0))
▸
|
✗
≡
|
Math.scale0(datacolumn)
bestimmt den Skalierungsdeskriptor für [0,1], der für ML benötigt wirdML.preprocess(data,format,options)
Funktion können nun alle Datenvariablen und Instanzen transformiert und sklaiert werdenxmy
(d.h. zwei x/y Matrizen und liefert {x:number [][], y:number [][]}
)1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
var self=this;
this.data=dataEE01;
// Alle Attribute der Datentabelle inkl. Zielattr.
this.attributes = ML.statistics.features(this.data);
this.targets = ['Y1','Y2'];
this.features = this.attributes.without(this.targets);
print(this.features)
// Berechnung der Skalierungsdiskreptoren
this.scales = this.attributes.map(function (attr) {
return Math.scale0(self.data.pluck(attr),0,1);
});
this.scalesX = this.scales.slice(0,this.features.length);
this.scalesY = this.scales.slice(this.features.length);
// Transformation und Skalierung der Datentabelle
this.dataP =
ML.preprocess(this.data,'xmy',
{features:this.features,target:this.targets,
xscale:this.scalesX,
yscale:this.scalesY})
Table(this.dataP.x);
Table(this.dataP.y);
▸
|
✗
≡
|
Frage. Warum ist die automatische Skalierung der Trainings- und Testdaten evtl. problematisch?
ML.unscale
Funktion wieder expandiert werden:1
2
3
4
5
var self=this;
var yUnscaled = this.dataP.y.map(function (row) {
return ML.unscale(row,self.scalesY);
});
Table(yUnscaled);
▸
|
✗
≡
|
1
2
3
4
5
6
7
8
9
10
11
12
13
var x = [
[0,0],
[0,1],
[1,0],
[1,1]
]
var y = [
[0],
[1],
[1],
[0]
]
this.dataEXOR={x:x,y:y}
▸
|
✗
≡
|
1
2
3
4
5
6
this.modelEXOR=ML.learner({
algorithm: ML.ML.MLP,
layers : [2,4,1],
verbose: 1,
});
ML.log(print);
▸
|
✗
≡
|
1
2
3
4
5
6
ML.train(this.modelEXOR,{
x:this.dataEXOR.x,
y:this.dataEXOR.y,
epochs:1000
});
print(this.modelEXOR.time+' ms');
▸
|
✗
≡
|
1
2
3
var results=ML.predict(this.modelEXOR,this.dataEXOR.x);
print(results)
print(ML.statistics.crossEntropy(ML.math.vec2Mat(results),this.dataEXOR.y));
▸
|
✗
≡
|
Punkte: | 1. | /2 | 2. | /2 | 3. | /2 | 4. | /2 |
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
var self=this;
this.data=dataEE01;
this.attributes = ML.statistics.features(this.data);
this.targets = ['Y1','Y2'];
this.features = this.attributes.without(this.targets);
print(this.features)
// Berechnung der Skalierungsdeskriptoren
this.scales = this.attributes.map(function (attr) {
return Math.scale0(self.data.pluck(attr),0,1);
});
this.scalesX = this.scales.slice(0,this.features.length);
this.scalesY = this.scales.slice(this.features.length);
// Transformation und Skalierung
this.dataP =
ML.preprocess(this.data,'xmy',
{features:this.features,target:this.targets,
xscale:this.scalesX,
yscale:this.scalesY})
▸
|
✗
≡
|
Benutze D=
this.data
und führe bei Datenteilung wieder die Transformation und Skalierung durch (siehe oben).
Versuche zunächst ein Modell mit den zwei Zielvariablen Y1 und Y2. Wenn es keine gute Konvergenz beim Training gibt können auch zwei getrennte Modelle für Y1 und Y2 erzeugt und trainiert werden.
this.model
). Wähle mindestens drei Konfigurationen (Variation der inneren Schichten des ANN, auch ohne innere Schicht)Eine Kodierung von numerischen Werten ist nicht erforderlich!
Im folgenden Eingabefeld die Ergebnisse zusammentragen (geeignete Features, verschiedene Versuche, Fehler, Rechenzeit für das Training, Inferenzzeit beim Test):
Alle Elemente von der this
Variable werden in der JSON Datei gespeichert und können wieder geladen werden (also Daten, Modelle usw).
Die Rechenzeiten für das Training können schon empfindlich ansteigen (bei ca. 400 Instanzen ungefähr 1s pro 1000 Epochen)
Eventuell werden mehr als 100000 Trainingsepochen benötigt (sollte partitioniert ausgeführt werden, nicht auf einmal durchführen)
Der letzte Trainingsfehler (mean squared error) und die Kreuzentropie können über this.model.error
und this.model.crossEntropy
abgerufen werden.
1
2
3
4
var data = this.data; // preprocess nach Teilung erforderlich!
// ML.split, siehe oben
this.test1 = ..;
this.train1 = ..;
▸
|
✗
≡
|
1
2
3
this.model=ML.learner({
// TODO, Vorschlag für Konf.: [8,4,2,2]
});
▸
|
✗
≡
|
1
2
3
ML.train(this.model,{
// TODO
});
▸
|
✗
≡
|
1
// this.model, this.train1 und this.test1 verwenden, siehe oben
▸
|
✗
≡
|