Logo ROOT   6.16/01
Reference Guide
Adam.h
Go to the documentation of this file.
1// @(#)root/tmva/tmva/dnn:$Id$
2// Author: Ravi Kiran S
3
4/**********************************************************************************
5 * Project: TMVA - a Root-integrated toolkit for multivariate data analysis *
6 * Package: TMVA *
7 * Class : TAdam *
8 * Web : http://tmva.sourceforge.net *
9 * *
10 * Description: *
11 * Adam Optimizer Class *
12 * *
13 * Authors (alphabetical): *
14 * Ravi Kiran S <sravikiran0606@gmail.com> - CERN, Switzerland *
15 * *
16 * Copyright (c) 2005-2018: *
17 * CERN, Switzerland *
18 * U. of Victoria, Canada *
19 * MPI-K Heidelberg, Germany *
20 * U. of Bonn, Germany *
21 * *
22 * Redistribution and use in source and binary forms, with or without *
23 * modification, are permitted according to the terms listed in LICENSE *
24 * (http://tmva.sourceforge.net/LICENSE) *
25 **********************************************************************************/
26
27#ifndef TMVA_DNN_ADAM
28#define TMVA_DNN_ADAM
29
30#include "TMatrix.h"
31#include "TMVA/DNN/Optimizer.h"
32#include "TMVA/DNN/Functions.h"
33
34namespace TMVA {
35namespace DNN {
36
37/** \class TAdam
38 * Adam Optimizer class
39 *
40 * This class represents the Adam Optimizer.
41 */
42template <typename Architecture_t, typename Layer_t = VGeneralLayer<Architecture_t>,
43 typename DeepNet_t = TDeepNet<Architecture_t, Layer_t>>
44class TAdam : public VOptimizer<Architecture_t, Layer_t, DeepNet_t> {
45public:
46 using Matrix_t = typename Architecture_t::Matrix_t;
47 using Scalar_t = typename Architecture_t::Scalar_t;
48
49protected:
50 Scalar_t fBeta1; ///< The Beta1 constant used by the optimizer.
51 Scalar_t fBeta2; ///< The Beta2 constant used by the optimizer.
52 Scalar_t fEpsilon; ///< The Smoothing term used to avoid division by zero.
53
54 std::vector<std::vector<Matrix_t>> fFirstMomentWeights; ///< The decaying average of the first moment of the past
55 /// weight gradients associated with the deep net.
56 std::vector<std::vector<Matrix_t>> fFirstMomentBiases; ///< The decaying average of the first moment of the past bias
57 /// gradients associated with the deep net.
58
59 std::vector<std::vector<Matrix_t>> fSecondMomentWeights; ///< The decaying average of the second moment of the past
60 /// weight gradients associated with the deep net.
61 std::vector<std::vector<Matrix_t>> fSecondMomentBiases; ///< The decaying average of the second moment of the past
62 /// bias gradients associated with the deep net.
63
64 /*! Update the weights, given the current weight gradients. */
65 void UpdateWeights(size_t layerIndex, std::vector<Matrix_t> &weights, const std::vector<Matrix_t> &weightGradients);
66
67 /*! Update the biases, given the current bias gradients. */
68 void UpdateBiases(size_t layerIndex, std::vector<Matrix_t> &biases, const std::vector<Matrix_t> &biasGradients);
69
70public:
71 /*! Constructor. */
72 TAdam(DeepNet_t &deepNet, Scalar_t learningRate = 0.001, Scalar_t beta1 = 0.9, Scalar_t beta2 = 0.999,
73 Scalar_t epsilon = 1e-8);
74
75 /*! Destructor. */
76 ~TAdam() = default;
77
78 /*! Getters */
79 Scalar_t GetBeta1() const { return fBeta1; }
80 Scalar_t GetBeta2() const { return fBeta2; }
81 Scalar_t GetEpsilon() const { return fEpsilon; }
82
83 std::vector<std::vector<Matrix_t>> &GetFirstMomentWeights() { return fFirstMomentWeights; }
84 std::vector<Matrix_t> &GetFirstMomentWeightsAt(size_t i) { return fFirstMomentWeights[i]; }
85
86 std::vector<std::vector<Matrix_t>> &GetFirstMomentBiases() { return fFirstMomentBiases; }
87 std::vector<Matrix_t> &GetFirstMomentBiasesAt(size_t i) { return fFirstMomentBiases[i]; }
88
89 std::vector<std::vector<Matrix_t>> &GetSecondMomentWeights() { return fSecondMomentWeights; }
90 std::vector<Matrix_t> &GetSecondMomentWeightsAt(size_t i) { return fSecondMomentWeights[i]; }
91
92 std::vector<std::vector<Matrix_t>> &GetSecondMomentBiases() { return fSecondMomentBiases; }
93 std::vector<Matrix_t> &GetSecondMomentBiasesAt(size_t i) { return fSecondMomentBiases[i]; }
94};
95
96//
97//
98// The Adam Optimizer Class - Implementation
99//_________________________________________________________________________________________________
100template <typename Architecture_t, typename Layer_t, typename DeepNet_t>
103 : VOptimizer<Architecture_t, Layer_t, DeepNet_t>(learningRate, deepNet), fBeta1(beta1), fBeta2(beta2),
104 fEpsilon(epsilon)
105{
106 std::vector<Layer_t *> &layers = deepNet.GetLayers();
107 const size_t layersNSlices = layers.size();
108 fFirstMomentWeights.resize(layersNSlices);
109 fFirstMomentBiases.resize(layersNSlices);
110 fSecondMomentWeights.resize(layersNSlices);
111 fSecondMomentBiases.resize(layersNSlices);
112
113 for (size_t i = 0; i < layersNSlices; i++) {
114 const size_t weightsNSlices = (layers[i]->GetWeights()).size();
115
116 for (size_t j = 0; j < weightsNSlices; j++) {
117 Matrix_t &currentWeights = layers[i]->GetWeightsAt(j);
118 const size_t weightsNRows = currentWeights.GetNrows();
119 const size_t weightsNCols = currentWeights.GetNcols();
120
121 fFirstMomentWeights[i].emplace_back(weightsNRows, weightsNCols);
122 fSecondMomentWeights[i].emplace_back(weightsNRows, weightsNCols);
123 initialize<Architecture_t>(fFirstMomentWeights[i][j], EInitialization::kZero);
124 initialize<Architecture_t>(fSecondMomentWeights[i][j], EInitialization::kZero);
125 }
126
127 const size_t biasesNSlices = (layers[i]->GetBiases()).size();
128
129 for (size_t j = 0; j < biasesNSlices; j++) {
130 Matrix_t &currentBiases = layers[i]->GetBiasesAt(j);
131 const size_t biasesNRows = currentBiases.GetNrows();
132 const size_t biasesNCols = currentBiases.GetNcols();
133
134 fFirstMomentBiases[i].emplace_back(biasesNRows, biasesNCols);
135 fSecondMomentBiases[i].emplace_back(biasesNRows, biasesNCols);
136 initialize<Architecture_t>(fFirstMomentBiases[i][j], EInitialization::kZero);
137 initialize<Architecture_t>(fSecondMomentBiases[i][j], EInitialization::kZero);
138 }
139 }
140}
141
142//_________________________________________________________________________________________________
143template <typename Architecture_t, typename Layer_t, typename DeepNet_t>
144auto TAdam<Architecture_t, Layer_t, DeepNet_t>::UpdateWeights(size_t layerIndex, std::vector<Matrix_t> &weights,
145 const std::vector<Matrix_t> &weightGradients) -> void
146{
147 std::vector<Matrix_t> &currentLayerFirstMomentWeights = this->GetFirstMomentWeightsAt(layerIndex);
148 std::vector<Matrix_t> &currentLayerSecondMomentWeights = this->GetSecondMomentWeightsAt(layerIndex);
149
150 // alpha = learningRate * sqrt(1 - beta2^t) / (1-beta1^t)
151 Scalar_t alpha = (this->GetLearningRate()) * (sqrt(1 - pow(this->GetBeta2(), this->GetGlobalStep()))) /
152 (1 - pow(this->GetBeta1(), this->GetGlobalStep()));
153
154 /// Adam update of first and second momentum of the weights
155 for (size_t i = 0; i < weights.size(); i++) {
156 // Mt = beta1 * Mt-1 + (1-beta1) * WeightGradients
157 Architecture_t::AdamUpdateFirstMom(currentLayerFirstMomentWeights[i], weightGradients[i], this->GetBeta1() );
158 // Vt = beta2 * Vt-1 + (1-beta2) * WeightGradients^2
159 Architecture_t::AdamUpdateSecondMom(currentLayerSecondMomentWeights[i], weightGradients[i], this->GetBeta2() );
160 // Weight = Weight - alpha * Mt / (sqrt(Vt) + epsilon)
161 Architecture_t::AdamUpdate(weights[i], currentLayerFirstMomentWeights[i], currentLayerSecondMomentWeights[i],
162 alpha, this->GetEpsilon() );
163 }
164}
165
166//_________________________________________________________________________________________________
167template <typename Architecture_t, typename Layer_t, typename DeepNet_t>
168auto TAdam<Architecture_t, Layer_t, DeepNet_t>::UpdateBiases(size_t layerIndex, std::vector<Matrix_t> &biases,
169 const std::vector<Matrix_t> &biasGradients) -> void
170{
171 std::vector<Matrix_t> &currentLayerFirstMomentBiases = this->GetFirstMomentBiasesAt(layerIndex);
172 std::vector<Matrix_t> &currentLayerSecondMomentBiases = this->GetSecondMomentBiasesAt(layerIndex);
173
174 // alpha = learningRate * sqrt(1 - beta2^t) / (1-beta1^t)
175 Scalar_t alpha = (this->GetLearningRate()) * (sqrt(1 - pow(this->GetBeta2(), this->GetGlobalStep()))) /
176 (1 - pow(this->GetBeta1(), this->GetGlobalStep()));
177
178 // updating of the biases.
179 for (size_t i = 0; i < biases.size(); i++) {
180 // Mt = beta1 * Mt-1 + (1-beta1) * BiasGradients
181 Architecture_t::AdamUpdateFirstMom(currentLayerFirstMomentBiases[i], biasGradients[i], this->GetBeta1() );
182 // Vt = beta2 * Vt-1 + (1-beta2) * BiasGradients^2
183 Architecture_t::AdamUpdateSecondMom(currentLayerSecondMomentBiases[i], biasGradients[i], this->GetBeta2() );
184 // theta = theta - alpha * Mt / (sqrt(Vt) + epsilon)
185 Architecture_t::AdamUpdate(biases[i], currentLayerFirstMomentBiases[i], currentLayerSecondMomentBiases[i],
186 alpha, this->GetEpsilon() );
187 }
188}
189
190} // namespace DNN
191} // namespace TMVA
192
193#endif
#define e(i)
Definition: RSha256.hxx:103
double pow(double, double)
double sqrt(double)
Adam Optimizer class.
Definition: Adam.h:44
std::vector< std::vector< Matrix_t > > fSecondMomentWeights
The decaying average of the second moment of the past weight gradients associated with the deep net.
Definition: Adam.h:59
std::vector< Matrix_t > & GetSecondMomentBiasesAt(size_t i)
Definition: Adam.h:93
Scalar_t GetEpsilon() const
Definition: Adam.h:81
typename Architecture_t::Matrix_t Matrix_t
Definition: Adam.h:46
std::vector< Matrix_t > & GetFirstMomentBiasesAt(size_t i)
Definition: Adam.h:87
Scalar_t fBeta2
The Beta2 constant used by the optimizer.
Definition: Adam.h:51
std::vector< std::vector< Matrix_t > > fSecondMomentBiases
The decaying average of the second moment of the past bias gradients associated with the deep net.
Definition: Adam.h:61
std::vector< std::vector< Matrix_t > > & GetSecondMomentBiases()
Definition: Adam.h:92
~TAdam()=default
Destructor.
std::vector< Matrix_t > & GetFirstMomentWeightsAt(size_t i)
Definition: Adam.h:84
std::vector< std::vector< Matrix_t > > & GetSecondMomentWeights()
Definition: Adam.h:89
TAdam(DeepNet_t &deepNet, Scalar_t learningRate=0.001, Scalar_t beta1=0.9, Scalar_t beta2=0.999, Scalar_t epsilon=1e-8)
Constructor.
Definition: Adam.h:101
Scalar_t GetBeta2() const
Definition: Adam.h:80
std::vector< std::vector< Matrix_t > > fFirstMomentBiases
The decaying average of the first moment of the past bias gradients associated with the deep net.
Definition: Adam.h:56
Scalar_t fEpsilon
The Smoothing term used to avoid division by zero.
Definition: Adam.h:52
void UpdateWeights(size_t layerIndex, std::vector< Matrix_t > &weights, const std::vector< Matrix_t > &weightGradients)
Update the weights, given the current weight gradients.
Definition: Adam.h:144
std::vector< std::vector< Matrix_t > > & GetFirstMomentWeights()
Definition: Adam.h:83
std::vector< Matrix_t > & GetSecondMomentWeightsAt(size_t i)
Definition: Adam.h:90
std::vector< std::vector< Matrix_t > > fFirstMomentWeights
The decaying average of the first moment of the past weight gradients associated with the deep net.
Definition: Adam.h:54
Scalar_t fBeta1
The Beta1 constant used by the optimizer.
Definition: Adam.h:50
void UpdateBiases(size_t layerIndex, std::vector< Matrix_t > &biases, const std::vector< Matrix_t > &biasGradients)
Update the biases, given the current bias gradients.
Definition: Adam.h:168
std::vector< std::vector< Matrix_t > > & GetFirstMomentBiases()
Definition: Adam.h:86
typename Architecture_t::Scalar_t Scalar_t
Definition: Adam.h:47
Scalar_t GetBeta1() const
Getters.
Definition: Adam.h:79
Generic Optimizer class.
Definition: Optimizer.h:44
std::vector< Layer_t * > & GetLayers()
Definition: Optimizer.h:78
Abstract ClassifierFactory template that handles arbitrary types.
REAL epsilon
Definition: triangle.c:617