regression_metric.hpp 5.11 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 override {
52
    double sum_loss = 0.0f;
53
54
55
56
    if (weights_ == nullptr) {
#pragma omp parallel for schedule(static) reduction(+:sum_loss)
      for (data_size_t i = 0; i < num_data_; ++i) {
        // add loss
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
57
        sum_loss += PointWiseLossCalculator::LossOnPoint(label_[i], score[i], huber_delta_, fair_c_);
Guolin Ke's avatar
Guolin Ke committed
58
      }
59
60
61
62
    } else {
#pragma omp parallel for schedule(static) reduction(+:sum_loss)
      for (data_size_t i = 0; i < num_data_; ++i) {
        // add loss
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
63
        sum_loss += PointWiseLossCalculator::LossOnPoint(label_[i], score[i], huber_delta_, fair_c_) * weights_[i];
wxchan's avatar
wxchan committed
64
      }
Guolin Ke's avatar
Guolin Ke committed
65
    }
66
67
    double loss = PointWiseLossCalculator::AverageLoss(sum_loss, sum_weights_);
    return std::vector<double>(1, loss);
68

Guolin Ke's avatar
Guolin Ke committed
69
70
  }

71
  inline static double AverageLoss(double sum_loss, double sum_weights) {
Guolin Ke's avatar
Guolin Ke committed
72
73
74
    return sum_loss / sum_weights;
  }

Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
75
76
protected:
  /*! \brief delta for Huber loss */
77
  double huber_delta_;
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
78
  /*! \brief c for Fair loss */
79
  double fair_c_;
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
80

Guolin Ke's avatar
Guolin Ke committed
81
82
83
84
85
86
87
88
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 */
89
  double sum_weights_;
Guolin Ke's avatar
Guolin Ke committed
90
  /*! \brief Name of this test set */
91
  std::vector<std::string> name_;
Guolin Ke's avatar
Guolin Ke committed
92
93
94
95
96
97
98
};

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

99
  inline static double LossOnPoint(float label, double score, double, double) {
Guolin Ke's avatar
Guolin Ke committed
100
101
102
    return (score - label)*(score - label);
  }

103
  inline static double AverageLoss(double sum_loss, double sum_weights) {
Guolin Ke's avatar
Guolin Ke committed
104
105
106
107
108
    // need sqrt the result for L2 loss
    return std::sqrt(sum_loss / sum_weights);
  }

  inline static const char* Name() {
109
    return "l2";
Guolin Ke's avatar
Guolin Ke committed
110
111
112
113
114
115
116
117
  }
};

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

118
  inline static double LossOnPoint(float label, double score, double, double) {
Guolin Ke's avatar
Guolin Ke committed
119
120
121
    return std::fabs(score - label);
  }
  inline static const char* Name() {
122
    return "l1";
Guolin Ke's avatar
Guolin Ke committed
123
124
125
  }
};

Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
126
127
128
/*! \brief Huber loss for regression task */
class HuberLossMetric: public RegressionMetric<HuberLossMetric> {
public:
Guolin Ke's avatar
Guolin Ke committed
129
  explicit HuberLossMetric(const MetricConfig& config) :RegressionMetric<HuberLossMetric>(config) {
130
    huber_delta_ = static_cast<double>(config.huber_delta);
Guolin Ke's avatar
Guolin Ke committed
131
  }
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
132

133
134
  inline static double LossOnPoint(float label, double score, double delta, double) {
    const double diff = score - label;
Guolin Ke's avatar
Guolin Ke committed
135
136
137
138
    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
139
    }
Guolin Ke's avatar
Guolin Ke committed
140
  }
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
141

Guolin Ke's avatar
Guolin Ke committed
142
143
144
  inline static const char* Name() {
    return "huber";
  }
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
145
146
};

Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
147
148
149
150
/*! \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
151
  explicit FairLossMetric(const MetricConfig& config) :RegressionMetric<FairLossMetric>(config) {
152
    fair_c_ = static_cast<double>(config.fair_c);
Guolin Ke's avatar
Guolin Ke committed
153
  }
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
154

155
156
  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
157
158
    return c * x - c * c * std::log(1.0f + x / c);
  }
Tsukasa OMOTO's avatar
Tsukasa OMOTO committed
159

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

165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
/*! \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
183
}  // namespace LightGBM
Guolin Ke's avatar
Guolin Ke committed
184
#endif   // LightGBM_METRIC_REGRESSION_METRIC_HPP_