regression_metric.hpp 5.87 KB
Newer Older
Guolin Ke's avatar
Guolin Ke committed
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
#ifndef LIGHTGBM_METRIC_REGRESSION_METRIC_HPP_
#define LIGHTGBM_METRIC_REGRESSION_METRIC_HPP_

#include <LightGBM/utils/log.h>

#include <LightGBM/metric.h>

#include <cmath>

namespace LightGBM {
/*!
* \brief Metric for regression task.
* Use static class "PointWiseLossCalculator" to calculate loss point-wise
*/
template<typename PointWiseLossCalculator>
class RegressionMetric: public Metric {
public:
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
18
  explicit RegressionMetric(const MetricConfig&) :huber_delta_(1.0f), fair_c_(1.0f) {
Guolin Ke's avatar
Guolin Ke committed
19
20
21
22
23
24
  }

  virtual ~RegressionMetric() {

  }

Guolin Ke's avatar
Guolin Ke committed
25
  const std::vector<std::string>& GetName() const override {
26
    return name_;
27
28
  }

29
  double factor_to_bigger_better() const override {
30
    return -1.0f;
31
32
  }

Guolin Ke's avatar
Guolin Ke committed
33
34
  void Init(const Metadata& metadata, data_size_t num_data) override {
    name_.emplace_back(PointWiseLossCalculator::Name());
35

Guolin Ke's avatar
Guolin Ke committed
36
37
38
39
40
41
    num_data_ = num_data;
    // get label
    label_ = metadata.label();
    // get weights
    weights_ = metadata.weights();
    if (weights_ == nullptr) {
42
      sum_weights_ = static_cast<double>(num_data_);
Guolin Ke's avatar
Guolin Ke committed
43
44
45
46
47
48
49
    } else {
      sum_weights_ = 0.0f;
      for (data_size_t i = 0; i < num_data_; ++i) {
        sum_weights_ += weights_[i];
      }
    }
  }
50

51
  std::vector<double> Eval(const double* score, const ObjectiveFunction* objective, int) const override {
52
    double sum_loss = 0.0f;
53
54
55
56
57
58
59
60
61
62
63
64
65
    if (objective == nullptr) {
      if (weights_ == nullptr) {
        #pragma omp parallel for schedule(static) reduction(+:sum_loss)
        for (data_size_t i = 0; i < num_data_; ++i) {
          // add loss
          sum_loss += PointWiseLossCalculator::LossOnPoint(label_[i], score[i], huber_delta_, fair_c_);
        }
      } else {
        #pragma omp parallel for schedule(static) reduction(+:sum_loss)
        for (data_size_t i = 0; i < num_data_; ++i) {
          // add loss
          sum_loss += PointWiseLossCalculator::LossOnPoint(label_[i], score[i], huber_delta_, fair_c_) * weights_[i];
        }
Guolin Ke's avatar
Guolin Ke committed
66
      }
67
    } else {
68
69
70
71
72
73
74
75
76
77
78
79
      if (weights_ == nullptr) {
        #pragma omp parallel for schedule(static) reduction(+:sum_loss)
        for (data_size_t i = 0; i < num_data_; ++i) {
          // add loss
          sum_loss += PointWiseLossCalculator::LossOnPoint(label_[i], objective->ConvertOutput(score[i]), huber_delta_, fair_c_);
        }
      } else {
        #pragma omp parallel for schedule(static) reduction(+:sum_loss)
        for (data_size_t i = 0; i < num_data_; ++i) {
          // add loss
          sum_loss += PointWiseLossCalculator::LossOnPoint(label_[i], objective->ConvertOutput(score[i]), huber_delta_, fair_c_) * weights_[i];
        }
wxchan's avatar
wxchan committed
80
      }
Guolin Ke's avatar
Guolin Ke committed
81
    }
82
83
    double loss = PointWiseLossCalculator::AverageLoss(sum_loss, sum_weights_);
    return std::vector<double>(1, loss);
84

Guolin Ke's avatar
Guolin Ke committed
85
86
  }

87
  inline static double AverageLoss(double sum_loss, double sum_weights) {
Guolin Ke's avatar
Guolin Ke committed
88
89
90
    return sum_loss / sum_weights;
  }

Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
91
92
protected:
  /*! \brief delta for Huber loss */
93
  double huber_delta_;
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
94
  /*! \brief c for Fair loss */
95
  double fair_c_;
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
96

Guolin Ke's avatar
Guolin Ke committed
97
98
99
100
101
102
103
104
private:
  /*! \brief Number of data */
  data_size_t num_data_;
  /*! \brief Pointer of label */
  const float* label_;
  /*! \brief Pointer of weighs */
  const float* weights_;
  /*! \brief Sum weights */
105
  double sum_weights_;
Guolin Ke's avatar
Guolin Ke committed
106
  /*! \brief Name of this test set */
107
  std::vector<std::string> name_;
Guolin Ke's avatar
Guolin Ke committed
108
109
110
111
112
113
114
};

/*! \brief L2 loss for regression task */
class L2Metric: public RegressionMetric<L2Metric> {
public:
  explicit L2Metric(const MetricConfig& config) :RegressionMetric<L2Metric>(config) {}

115
  inline static double LossOnPoint(float label, double score, double, double) {
Guolin Ke's avatar
Guolin Ke committed
116
117
118
    return (score - label)*(score - label);
  }

119
  inline static double AverageLoss(double sum_loss, double sum_weights) {
Guolin Ke's avatar
Guolin Ke committed
120
121
122
123
124
    // need sqrt the result for L2 loss
    return std::sqrt(sum_loss / sum_weights);
  }

  inline static const char* Name() {
125
    return "l2";
Guolin Ke's avatar
Guolin Ke committed
126
127
128
129
130
131
132
133
  }
};

/*! \brief L1 loss for regression task */
class L1Metric: public RegressionMetric<L1Metric> {
public:
  explicit L1Metric(const MetricConfig& config) :RegressionMetric<L1Metric>(config) {}

134
  inline static double LossOnPoint(float label, double score, double, double) {
Guolin Ke's avatar
Guolin Ke committed
135
136
137
    return std::fabs(score - label);
  }
  inline static const char* Name() {
138
    return "l1";
Guolin Ke's avatar
Guolin Ke committed
139
140
141
  }
};

Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
142
143
144
/*! \brief Huber loss for regression task */
class HuberLossMetric: public RegressionMetric<HuberLossMetric> {
public:
Guolin Ke's avatar
Guolin Ke committed
145
  explicit HuberLossMetric(const MetricConfig& config) :RegressionMetric<HuberLossMetric>(config) {
146
    huber_delta_ = static_cast<double>(config.huber_delta);
Guolin Ke's avatar
Guolin Ke committed
147
  }
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
148

149
150
  inline static double LossOnPoint(float label, double score, double delta, double) {
    const double diff = score - label;
Guolin Ke's avatar
Guolin Ke committed
151
152
153
154
    if (std::abs(diff) <= delta) {
      return 0.5f * diff * diff;
    } else {
      return delta * (std::abs(diff) - 0.5f * delta);
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
155
    }
Guolin Ke's avatar
Guolin Ke committed
156
  }
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
157

Guolin Ke's avatar
Guolin Ke committed
158
159
160
  inline static const char* Name() {
    return "huber";
  }
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
161
162
};

Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
163
164
165
166
/*! \brief Fair loss for regression task */
// http://research.microsoft.com/en-us/um/people/zhang/INRIA/Publis/Tutorial-Estim/node24.html
class FairLossMetric: public RegressionMetric<FairLossMetric> {
public:
Guolin Ke's avatar
Guolin Ke committed
167
  explicit FairLossMetric(const MetricConfig& config) :RegressionMetric<FairLossMetric>(config) {
168
    fair_c_ = static_cast<double>(config.fair_c);
Guolin Ke's avatar
Guolin Ke committed
169
  }
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
170

171
172
  inline static double LossOnPoint(float label, double score, double, double c) {
    const double x = std::fabs(score - label);
Guolin Ke's avatar
Guolin Ke committed
173
174
    return c * x - c * c * std::log(1.0f + x / c);
  }
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
175

Guolin Ke's avatar
Guolin Ke committed
176
177
178
  inline static const char* Name() {
    return "fair";
  }
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
179
180
};

181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
/*! \brief Poisson regression loss for regression task */
class PoissonMetric: public RegressionMetric<PoissonMetric> {
public:
  explicit PoissonMetric(const MetricConfig& config) :RegressionMetric<PoissonMetric>(config) {
  }

  inline static double LossOnPoint(float label, double score, double, double) {
    const double eps = 1e-10f;
    if (score < eps) {
      score = eps;
    }
    return score - label * std::log(score);
  }
  inline static const char* Name() {
    return "poisson";
  }
};

Guolin Ke's avatar
Guolin Ke committed
199
}  // namespace LightGBM
Guolin Ke's avatar
Guolin Ke committed
200
#endif   // LightGBM_METRIC_REGRESSION_METRIC_HPP_