dataset.h 21.6 KB
Newer Older
1
2
3
4
/*!
 * Copyright (c) 2016 Microsoft Corporation. All rights reserved.
 * Licensed under the MIT License. See LICENSE file in the project root for license information.
 */
Guolin Ke's avatar
Guolin Ke committed
5
6
#ifndef LIGHTGBM_DATASET_H_
#define LIGHTGBM_DATASET_H_
Guolin Ke's avatar
Guolin Ke committed
7

Guolin Ke's avatar
Guolin Ke committed
8
#include <LightGBM/config.h>
Guolin Ke's avatar
Guolin Ke committed
9
#include <LightGBM/feature_group.h>
10
#include <LightGBM/meta.h>
11
#include <LightGBM/utils/array_args.h>
12
#include <LightGBM/utils/common.h>
13
14
15
#include <LightGBM/utils/openmp_wrapper.h>
#include <LightGBM/utils/random.h>
#include <LightGBM/utils/text_reader.h>
Guolin Ke's avatar
Guolin Ke committed
16
17

#include <string>
18
19
#include <functional>
#include <memory>
20
#include <mutex>
21
22
23
#include <unordered_set>
#include <utility>
#include <vector>
Guolin Ke's avatar
Guolin Ke committed
24
25
26
27

namespace LightGBM {

/*! \brief forward declaration */
Guolin Ke's avatar
Guolin Ke committed
28
class DatasetLoader;
Guolin Ke's avatar
Guolin Ke committed
29
/*!
Hui Xue's avatar
Hui Xue committed
30
* \brief This class is used to store some meta(non-feature) data for training data,
31
*        e.g. labels, weights, initial scores, query level informations.
Guolin Ke's avatar
Guolin Ke committed
32
*
Qiwei Ye's avatar
Qiwei Ye committed
33
*        Some details:
34
*        1. Label, used for training.
Qiwei Ye's avatar
Qiwei Ye committed
35
36
*        2. Weights, weighs of records, optional
*        3. Query Boundaries, necessary for lambdarank.
37
38
39
40
*           The documents of i-th query is in [ query_boundaries[i], query_boundaries[i+1] )
*        4. Query Weights, auto calculate by weights and query_boundaries(if both of them are existed)
*           the weight for i-th query is sum(query_boundaries[i] , .., query_boundaries[i+1]) / (query_boundaries[i + 1] -  query_boundaries[i+1])
*        5. Initial score. optional. if existing, the model will boost from this score, otherwise will start from 0.
Guolin Ke's avatar
Guolin Ke committed
41
42
*/
class Metadata {
Nikita Titov's avatar
Nikita Titov committed
43
 public:
44
  /*!
45
  * \brief Null constructor
Guolin Ke's avatar
Guolin Ke committed
46
47
48
  */
  Metadata();
  /*!
49
  * \brief Initialization will load query level informations, since it is need for sampling data
Guolin Ke's avatar
Guolin Ke committed
50
51
52
  * \param data_filename Filename of data
  * \param init_score_filename Filename of initial score
  */
53
  void Init(const char* data_filename, const char* initscore_file);
Guolin Ke's avatar
Guolin Ke committed
54
  /*!
Guolin Ke's avatar
Guolin Ke committed
55
56
  * \brief init as subset
  * \param metadata Filename of data
57
  * \param used_indices
Guolin Ke's avatar
Guolin Ke committed
58
59
60
61
  * \param num_used_indices
  */
  void Init(const Metadata& metadata, const data_size_t* used_indices, data_size_t num_used_indices);
  /*!
Guolin Ke's avatar
Guolin Ke committed
62
63
64
65
66
67
68
69
  * \brief Initial with binary memory
  * \param memory Pointer to memory
  */
  void LoadFromMemory(const void* memory);
  /*! \brief Destructor */
  ~Metadata();

  /*!
Guolin Ke's avatar
Guolin Ke committed
70
  * \brief Initial work, will allocate space for label, weight(if exists) and query(if exists)
Guolin Ke's avatar
Guolin Ke committed
71
  * \param num_data Number of training data
Guolin Ke's avatar
Guolin Ke committed
72
73
  * \param weight_idx Index of weight column, < 0 means doesn't exists
  * \param query_idx Index of query id column, < 0 means doesn't exists
Guolin Ke's avatar
Guolin Ke committed
74
  */
75
  void Init(data_size_t num_data, int weight_idx, int query_idx);
Guolin Ke's avatar
Guolin Ke committed
76
77
78

  /*!
  * \brief Partition label by used indices
79
  * \param used_indices Indices of local used
Guolin Ke's avatar
Guolin Ke committed
80
81
82
83
84
85
86
87
88
  */
  void PartitionLabel(const std::vector<data_size_t>& used_indices);

  /*!
  * \brief Partition meta data according to local used indices if need
  * \param num_all_data Number of total training data, including other machines' data on parallel learning
  * \param used_data_indices Indices of local used training data
  */
  void CheckOrPartition(data_size_t num_all_data,
89
                        const std::vector<data_size_t>& used_data_indices);
Guolin Ke's avatar
Guolin Ke committed
90

91
  void SetLabel(const label_t* label, data_size_t len);
Guolin Ke's avatar
Guolin Ke committed
92

93
  void SetWeights(const label_t* weights, data_size_t len);
Guolin Ke's avatar
Guolin Ke committed
94

Guolin Ke's avatar
Guolin Ke committed
95
  void SetQuery(const data_size_t* query, data_size_t len);
Guolin Ke's avatar
Guolin Ke committed
96

Guolin Ke's avatar
Guolin Ke committed
97
98
99
100
  /*!
  * \brief Set initial scores
  * \param init_score Initial scores, this class will manage memory for init_score.
  */
101
  void SetInitScore(const double* init_score, data_size_t len);
Guolin Ke's avatar
Guolin Ke committed
102
103
104
105
106
107


  /*!
  * \brief Save binary data to file
  * \param file File want to write
  */
108
  void SaveBinaryToFile(const VirtualFileWriter* writer) const;
Guolin Ke's avatar
Guolin Ke committed
109
110
111
112
113
114
115
116
117
118

  /*!
  * \brief Get sizes in byte of this object
  */
  size_t SizesInByte() const;

  /*!
  * \brief Get pointer of label
  * \return Pointer of label
  */
119
  inline const label_t* label() const { return label_.data(); }
Guolin Ke's avatar
Guolin Ke committed
120
121
122
123
124
125

  /*!
  * \brief Set label for one record
  * \param idx Index of this record
  * \param value Label value of this record
  */
126
  inline void SetLabelAt(data_size_t idx, label_t value) {
127
    label_[idx] = value;
Guolin Ke's avatar
Guolin Ke committed
128
129
  }

Guolin Ke's avatar
Guolin Ke committed
130
131
132
133
134
  /*!
  * \brief Set Weight for one record
  * \param idx Index of this record
  * \param value Weight value of this record
  */
135
  inline void SetWeightAt(data_size_t idx, label_t value) {
136
    weights_[idx] = value;
Guolin Ke's avatar
Guolin Ke committed
137
138
139
140
141
142
143
  }

  /*!
  * \brief Set Query Id for one record
  * \param idx Index of this record
  * \param value Query Id value of this record
  */
144
  inline void SetQueryAt(data_size_t idx, data_size_t value) {
Guolin Ke's avatar
Guolin Ke committed
145
146
147
    queries_[idx] = static_cast<data_size_t>(value);
  }

Guolin Ke's avatar
Guolin Ke committed
148
  /*!
Hui Xue's avatar
Hui Xue committed
149
  * \brief Get weights, if not exists, will return nullptr
Guolin Ke's avatar
Guolin Ke committed
150
151
  * \return Pointer of weights
  */
152
  inline const label_t* weights() const {
Guolin Ke's avatar
Guolin Ke committed
153
    if (!weights_.empty()) {
Guolin Ke's avatar
Guolin Ke committed
154
155
156
157
158
      return weights_.data();
    } else {
      return nullptr;
    }
  }
Guolin Ke's avatar
Guolin Ke committed
159
160

  /*!
Hui Xue's avatar
Hui Xue committed
161
  * \brief Get data boundaries on queries, if not exists, will return nullptr
162
  *        we assume data will order by query,
Guolin Ke's avatar
Guolin Ke committed
163
164
165
166
  *        the interval of [query_boundaris[i], query_boundaris[i+1])
  *        is the data indices for query i.
  * \return Pointer of data boundaries on queries
  */
167
  inline const data_size_t* query_boundaries() const {
Guolin Ke's avatar
Guolin Ke committed
168
    if (!query_boundaries_.empty()) {
Guolin Ke's avatar
Guolin Ke committed
169
170
171
172
173
      return query_boundaries_.data();
    } else {
      return nullptr;
    }
  }
Guolin Ke's avatar
Guolin Ke committed
174
175
176
177
178

  /*!
  * \brief Get Number of queries
  * \return Number of queries
  */
179
  inline data_size_t num_queries() const { return num_queries_; }
Guolin Ke's avatar
Guolin Ke committed
180
181

  /*!
Hui Xue's avatar
Hui Xue committed
182
  * \brief Get weights for queries, if not exists, will return nullptr
Guolin Ke's avatar
Guolin Ke committed
183
184
  * \return Pointer of weights for queries
  */
185
  inline const label_t* query_weights() const {
Guolin Ke's avatar
Guolin Ke committed
186
    if (!query_weights_.empty()) {
Guolin Ke's avatar
Guolin Ke committed
187
188
189
190
191
      return query_weights_.data();
    } else {
      return nullptr;
    }
  }
Guolin Ke's avatar
Guolin Ke committed
192
193

  /*!
Hui Xue's avatar
Hui Xue committed
194
  * \brief Get initial scores, if not exists, will return nullptr
Guolin Ke's avatar
Guolin Ke committed
195
196
  * \return Pointer of initial scores
  */
197
  inline const double* init_score() const {
Guolin Ke's avatar
Guolin Ke committed
198
    if (!init_score_.empty()) {
Guolin Ke's avatar
Guolin Ke committed
199
200
201
202
203
      return init_score_.data();
    } else {
      return nullptr;
    }
  }
Guolin Ke's avatar
Guolin Ke committed
204

205
206
207
  /*!
  * \brief Get size of initial scores
  */
Guolin Ke's avatar
Guolin Ke committed
208
  inline int64_t num_init_score() const { return num_init_score_; }
209

Guolin Ke's avatar
Guolin Ke committed
210
211
212
213
  /*! \brief Disable copy */
  Metadata& operator=(const Metadata&) = delete;
  /*! \brief Disable copy */
  Metadata(const Metadata&) = delete;
Guolin Ke's avatar
Guolin Ke committed
214

Nikita Titov's avatar
Nikita Titov committed
215
 private:
Guolin Ke's avatar
Guolin Ke committed
216
  /*! \brief Load initial scores from file */
217
  void LoadInitialScore(const char* initscore_file);
Guolin Ke's avatar
Guolin Ke committed
218
219
220
221
222
223
224
  /*! \brief Load wights from file */
  void LoadWeights();
  /*! \brief Load query boundaries from file */
  void LoadQueryBoundaries();
  /*! \brief Load query wights */
  void LoadQueryWeights();
  /*! \brief Filename of current data */
Guolin Ke's avatar
Guolin Ke committed
225
  std::string data_filename_;
Guolin Ke's avatar
Guolin Ke committed
226
227
228
229
230
  /*! \brief Number of data */
  data_size_t num_data_;
  /*! \brief Number of weights, used to check correct weight file */
  data_size_t num_weights_;
  /*! \brief Label data */
231
  std::vector<label_t> label_;
Guolin Ke's avatar
Guolin Ke committed
232
  /*! \brief Weights data */
233
  std::vector<label_t> weights_;
Guolin Ke's avatar
Guolin Ke committed
234
  /*! \brief Query boundaries */
Guolin Ke's avatar
Guolin Ke committed
235
  std::vector<data_size_t> query_boundaries_;
Guolin Ke's avatar
Guolin Ke committed
236
  /*! \brief Query weights */
237
  std::vector<label_t> query_weights_;
Guolin Ke's avatar
Guolin Ke committed
238
239
240
  /*! \brief Number of querys */
  data_size_t num_queries_;
  /*! \brief Number of Initial score, used to check correct weight file */
Guolin Ke's avatar
Guolin Ke committed
241
  int64_t num_init_score_;
Guolin Ke's avatar
Guolin Ke committed
242
  /*! \brief Initial score */
Guolin Ke's avatar
Guolin Ke committed
243
  std::vector<double> init_score_;
Guolin Ke's avatar
Guolin Ke committed
244
  /*! \brief Queries data */
Guolin Ke's avatar
Guolin Ke committed
245
  std::vector<data_size_t> queries_;
246
247
  /*! \brief mutex for threading safe call */
  std::mutex mutex_;
248
249
250
  bool weight_load_from_file_;
  bool query_load_from_file_;
  bool init_score_load_from_file_;
Guolin Ke's avatar
Guolin Ke committed
251
252
253
254
255
};


/*! \brief Interface for Parser */
class Parser {
Nikita Titov's avatar
Nikita Titov committed
256
 public:
Guolin Ke's avatar
Guolin Ke committed
257
258
259
260
261
262
  /*! \brief virtual destructor */
  virtual ~Parser() {}

  /*!
  * \brief Parse one line with label
  * \param str One line record, string format, should end with '\0'
Guolin Ke's avatar
Guolin Ke committed
263
264
  * \param out_features Output columns, store in (column_idx, values)
  * \param out_label Label will store to this if exists
Guolin Ke's avatar
Guolin Ke committed
265
266
  */
  virtual void ParseOneLine(const char* str,
267
                            std::vector<std::pair<int, double>>* out_features, double* out_label) const = 0;
Guolin Ke's avatar
Guolin Ke committed
268

269
  virtual int NumFeatures() const = 0;
Guolin Ke's avatar
Guolin Ke committed
270

Guolin Ke's avatar
Guolin Ke committed
271
  /*!
272
  * \brief Create an object of parser, will auto choose the format depend on file
Guolin Ke's avatar
Guolin Ke committed
273
  * \param filename One Filename of data
274
  * \param num_features Pass num_features of this data file if you know, <=0 means don't know
Guolin Ke's avatar
Guolin Ke committed
275
  * \param label_idx index of label column
Guolin Ke's avatar
Guolin Ke committed
276
277
  * \return Object of parser
  */
Guolin Ke's avatar
Guolin Ke committed
278
  static Parser* CreateParser(const char* filename, bool header, int num_features, int label_idx);
Guolin Ke's avatar
Guolin Ke committed
279
280
281
};

/*! \brief The main class of data set,
282
*          which are used to training or validation
Guolin Ke's avatar
Guolin Ke committed
283
284
*/
class Dataset {
Nikita Titov's avatar
Nikita Titov committed
285
 public:
Guolin Ke's avatar
Guolin Ke committed
286
  friend DatasetLoader;
Guolin Ke's avatar
Guolin Ke committed
287

288
  LIGHTGBM_EXPORT Dataset();
Guolin Ke's avatar
Guolin Ke committed
289

290
  LIGHTGBM_EXPORT Dataset(data_size_t num_data);
Guolin Ke's avatar
Guolin Ke committed
291

Guolin Ke's avatar
Guolin Ke committed
292
  void Construct(
Guolin Ke's avatar
Guolin Ke committed
293
    std::vector<std::unique_ptr<BinMapper>>* bin_mappers,
294
    int num_total_features,
295
    const std::vector<std::vector<double>>& forced_bins,
296
    int** sample_non_zero_indices,
Guolin Ke's avatar
Guolin Ke committed
297
    double** sample_values,
298
    const int* num_per_col,
299
    int num_sample_col,
Guolin Ke's avatar
Guolin Ke committed
300
    size_t total_sample_cnt,
Guolin Ke's avatar
Guolin Ke committed
301
    const Config& io_config);
Guolin Ke's avatar
Guolin Ke committed
302

Guolin Ke's avatar
Guolin Ke committed
303
  /*! \brief Destructor */
304
  LIGHTGBM_EXPORT ~Dataset();
Guolin Ke's avatar
Guolin Ke committed
305

306
  LIGHTGBM_EXPORT bool CheckAlign(const Dataset& other) const {
307
308
309
310
311
312
313
314
315
316
    if (num_features_ != other.num_features_) {
      return false;
    }
    if (num_total_features_ != other.num_total_features_) {
      return false;
    }
    if (label_idx_ != other.label_idx_) {
      return false;
    }
    for (int i = 0; i < num_features_; ++i) {
Guolin Ke's avatar
Guolin Ke committed
317
      if (!FeatureBinMapper(i)->CheckAlign(*(other.FeatureBinMapper(i)))) {
318
319
320
321
322
323
        return false;
      }
    }
    return true;
  }

Guolin Ke's avatar
Guolin Ke committed
324
325
326
327
328
329
330
331
332
333
  inline void FinishOneRow(int tid, data_size_t row_idx, const std::vector<bool>& is_feature_added) {
    if (is_finish_load_) { return; }
    for (auto fidx : feature_need_push_zeros_) {
      if (is_feature_added[fidx]) { continue; }
      const int group = feature2group_[fidx];
      const int sub_feature = feature2subfeature_[fidx];
      feature_groups_[group]->PushData(tid, sub_feature, row_idx, 0.0f);
    }
  }

Guolin Ke's avatar
Guolin Ke committed
334
  inline void PushOneRow(int tid, data_size_t row_idx, const std::vector<double>& feature_values) {
Guolin Ke's avatar
Guolin Ke committed
335
    if (is_finish_load_) { return; }
Guolin Ke's avatar
Guolin Ke committed
336
    for (size_t i = 0; i < feature_values.size() && i < static_cast<size_t>(num_total_features_); ++i) {
Guolin Ke's avatar
Guolin Ke committed
337
338
      int feature_idx = used_feature_map_[i];
      if (feature_idx >= 0) {
Guolin Ke's avatar
Guolin Ke committed
339
340
341
        const int group = feature2group_[feature_idx];
        const int sub_feature = feature2subfeature_[feature_idx];
        feature_groups_[group]->PushData(tid, sub_feature, row_idx, feature_values[i]);
Guolin Ke's avatar
Guolin Ke committed
342
343
344
345
      }
    }
  }

346
  inline void PushOneRow(int tid, data_size_t row_idx, const std::vector<std::pair<int, double>>& feature_values) {
Guolin Ke's avatar
Guolin Ke committed
347
    if (is_finish_load_) { return; }
Guolin Ke's avatar
Guolin Ke committed
348
    std::vector<bool> is_feature_added(num_features_, false);
349
    for (auto& inner_data : feature_values) {
350
      if (inner_data.first >= num_total_features_) { continue; }
351
352
      int feature_idx = used_feature_map_[inner_data.first];
      if (feature_idx >= 0) {
Guolin Ke's avatar
Guolin Ke committed
353
        is_feature_added[feature_idx] = true;
Guolin Ke's avatar
Guolin Ke committed
354
355
356
        const int group = feature2group_[feature_idx];
        const int sub_feature = feature2subfeature_[feature_idx];
        feature_groups_[group]->PushData(tid, sub_feature, row_idx, inner_data.second);
357
358
      }
    }
Guolin Ke's avatar
Guolin Ke committed
359
    FinishOneRow(tid, row_idx, is_feature_added);
360
361
  }

Guolin Ke's avatar
Guolin Ke committed
362
363
364
365
366
367
368
369
370
  inline void PushOneData(int tid, data_size_t row_idx, int group, int sub_feature, double value) {
    feature_groups_[group]->PushData(tid, sub_feature, row_idx, value);
  }

  inline int RealFeatureIndex(int fidx) const {
    return real_feature_idx_[fidx];
  }

  inline int InnerFeatureIndex(int col_idx) const {
Guolin Ke's avatar
Guolin Ke committed
371
    return used_feature_map_[col_idx];
Guolin Ke's avatar
Guolin Ke committed
372
  }
Guolin Ke's avatar
Guolin Ke committed
373
374
375
376
377
378
  inline int Feature2Group(int feature_idx) const {
    return feature2group_[feature_idx];
  }
  inline int Feture2SubFeature(int feature_idx) const {
    return feature2subfeature_[feature_idx];
  }
379
380
381
  inline uint64_t GroupBinBoundary(int group_idx) const {
    return group_bin_boundaries_[group_idx];
  }
Guolin Ke's avatar
Guolin Ke committed
382
383
384
  inline uint64_t NumTotalBin() const {
    return group_bin_boundaries_.back();
  }
385

386
387
388
389
390
391
392
393
394
  inline std::vector<int> ValidFeatureIndices() const {
    std::vector<int> ret;
    for (int i = 0; i < num_total_features_; ++i) {
      if (used_feature_map_[i] >= 0) {
        ret.push_back(i);
      }
    }
    return ret;
  }
Guolin Ke's avatar
Guolin Ke committed
395
396
397
  void ReSize(data_size_t num_data);

  void CopySubset(const Dataset* fullset, const data_size_t* used_indices, data_size_t num_used_indices, bool need_meta_data);
Guolin Ke's avatar
Guolin Ke committed
398

399
400
401
402
403
404
405
  MultiValBin* GetMultiBinFromSparseFeatures() const;

  MultiValBin* GetMultiBinFromAllFeatures() const;

  MultiValBin* TestMultiThreadingMethod(score_t* gradients, score_t* hessians, const std::vector<int8_t>& is_feature_used, bool is_constant_hessian,
    bool force_colwise, bool force_rowwise, bool* is_hist_col_wise) const;

406
  LIGHTGBM_EXPORT void FinishLoad();
Guolin Ke's avatar
Guolin Ke committed
407

408
  LIGHTGBM_EXPORT bool SetFloatField(const char* field_name, const float* field_data, data_size_t num_element);
Guolin Ke's avatar
Guolin Ke committed
409

410
  LIGHTGBM_EXPORT bool SetDoubleField(const char* field_name, const double* field_data, data_size_t num_element);
Guolin Ke's avatar
Guolin Ke committed
411

412
  LIGHTGBM_EXPORT bool SetIntField(const char* field_name, const int* field_data, data_size_t num_element);
413

414
  LIGHTGBM_EXPORT bool GetFloatField(const char* field_name, data_size_t* out_len, const float** out_ptr);
415

416
  LIGHTGBM_EXPORT bool GetDoubleField(const char* field_name, data_size_t* out_len, const double** out_ptr);
Guolin Ke's avatar
Guolin Ke committed
417

418
  LIGHTGBM_EXPORT bool GetIntField(const char* field_name, data_size_t* out_len, const int** out_ptr);
419

420
421
  LIGHTGBM_EXPORT bool GetInt8Field(const char* field_name, data_size_t* out_len, const int8_t** out_ptr);

Guolin Ke's avatar
Guolin Ke committed
422
423
424
  /*!
  * \brief Save current dataset into binary file, will save to "filename.bin"
  */
425
  LIGHTGBM_EXPORT void SaveBinaryFile(const char* bin_filename);
Guolin Ke's avatar
Guolin Ke committed
426

427
428
  LIGHTGBM_EXPORT void DumpTextFile(const char* text_filename);

429
  LIGHTGBM_EXPORT void CopyFeatureMapperFrom(const Dataset* dataset);
Guolin Ke's avatar
Guolin Ke committed
430

Guolin Ke's avatar
Guolin Ke committed
431
432
  LIGHTGBM_EXPORT void CreateValid(const Dataset* dataset);

433
434
  void ConstructHistograms(const std::vector<int8_t>& is_feature_used,
                           const data_size_t* data_indices, data_size_t num_data,
435
436
                           const score_t* gradients, const score_t* hessians,
                           score_t* ordered_gradients, score_t* ordered_hessians,
437
                           bool is_constant_hessian,
438
439
440
441
442
443
444
                           const MultiValBin* multi_val_bin, bool is_colwise,
                           hist_t* histogram_data) const;

  void ConstructHistogramsMultiVal(const MultiValBin* multi_val_bin, const data_size_t* data_indices, data_size_t num_data,
                                  const score_t* gradients, const score_t* hessians,
                                  bool is_constant_hessian,
                                  hist_t* histogram_data) const;
Guolin Ke's avatar
Guolin Ke committed
445

446
  void FixHistogram(int feature_idx, double sum_gradient, double sum_hessian, hist_t* data) const;
Guolin Ke's avatar
Guolin Ke committed
447

448
  inline data_size_t Split(int feature,
449
                           const uint32_t* threshold, int num_threshold,  bool default_left,
450
451
                           data_size_t* data_indices, data_size_t num_data,
                           data_size_t* lte_indices, data_size_t* gt_indices) const {
Guolin Ke's avatar
Guolin Ke committed
452
453
    const int group = feature2group_[feature];
    const int sub_feature = feature2subfeature_[feature];
454
    return feature_groups_[group]->Split(sub_feature, threshold, num_threshold, default_left, data_indices, num_data, lte_indices, gt_indices);
Guolin Ke's avatar
Guolin Ke committed
455
456
457
458
459
460
461
462
463
464
465
466
467
468
  }

  inline int SubFeatureBinOffset(int i) const {
    const int sub_feature = feature2subfeature_[i];
    if (sub_feature == 0) {
      return 1;
    } else {
      return 0;
    }
  }

  inline int FeatureNumBin(int i) const {
    const int group = feature2group_[i];
    const int sub_feature = feature2subfeature_[i];
469
    return feature_groups_[group]->bin_mappers_[sub_feature]->num_bin();
Guolin Ke's avatar
Guolin Ke committed
470
  }
Guolin Ke's avatar
Guolin Ke committed
471
472
473
474
475
476
477
478
479

  inline int8_t FeatureMonotone(int i) const {
    if (monotone_types_.empty()) {
      return 0;
    } else {
      return monotone_types_[i];
    }
  }

Guolin Ke's avatar
Guolin Ke committed
480
481
482
483
484
485
486
487
  inline double FeaturePenalte(int i) const {
    if (feature_penalty_.empty()) {
      return 1;
    } else {
      return feature_penalty_[i];
    }
  }

Guolin Ke's avatar
Guolin Ke committed
488
489
490
491
492
493
494
495
496
497
498
499
  bool HasMonotone() const {
    if (monotone_types_.empty()) {
      return false;
    } else {
      for (size_t i = 0; i < monotone_types_.size(); ++i) {
        if (monotone_types_[i] != 0) {
          return true;
        }
      }
      return false;
    }
  }
500

501
502
503
  inline int FeatureGroupNumBin(int group) const {
    return feature_groups_[group]->num_total_bin_;
  }
504

Guolin Ke's avatar
Guolin Ke committed
505
506
507
508
509
510
  inline const BinMapper* FeatureBinMapper(int i) const {
    const int group = feature2group_[i];
    const int sub_feature = feature2subfeature_[i];
    return feature_groups_[group]->bin_mappers_[sub_feature].get();
  }

511
512
513
514
  inline const Bin* FeatureGroupBin(int group) const {
    return feature_groups_[group]->bin_data_.get();
  }

Guolin Ke's avatar
Guolin Ke committed
515
516
517
  inline BinIterator* FeatureIterator(int i) const {
    const int group = feature2group_[i];
    const int sub_feature = feature2subfeature_[i];
zhangyafeikimi's avatar
zhangyafeikimi committed
518
    return feature_groups_[group]->SubFeatureIterator(sub_feature);
Guolin Ke's avatar
Guolin Ke committed
519
520
  }

521
522
523
  inline BinIterator* FeatureGroupIterator(int group) const {
    return feature_groups_[group]->FeatureGroupIterator();
  }
524

525
526
527
528
  inline bool IsMultiGroup(int i) const {
    return feature_groups_[i]->is_multi_val_;
  }

Guolin Ke's avatar
Guolin Ke committed
529
530
531
532
533
534
  inline double RealThreshold(int i, uint32_t threshold) const {
    const int group = feature2group_[i];
    const int sub_feature = feature2subfeature_[i];
    return feature_groups_[group]->bin_mappers_[sub_feature]->BinToValue(threshold);
  }

535
536
537
538
539
540
541
  // given a real threshold, find the closest threshold bin
  inline uint32_t BinThreshold(int i, double threshold_double) const {
    const int group = feature2group_[i];
    const int sub_feature = feature2subfeature_[i];
    return feature_groups_[group]->bin_mappers_[sub_feature]->ValueToBin(threshold_double);
  }

Guolin Ke's avatar
Guolin Ke committed
542
543
544
545
546
547
548
549
550
  /*!
  * \brief Get meta data pointer
  * \return Pointer of meta data
  */
  inline const Metadata& metadata() const { return metadata_; }

  /*! \brief Get Number of used features */
  inline int num_features() const { return num_features_; }

551
552
553
  /*! \brief Get Number of feature groups */
  inline int num_feature_groups() const { return num_groups_;}

554
555
556
  /*! \brief Get Number of total features */
  inline int num_total_features() const { return num_total_features_; }

Guolin Ke's avatar
Guolin Ke committed
557
558
559
560
  /*! \brief Get the index of label column */
  inline int label_idx() const { return label_idx_; }

  /*! \brief Get names of current data set */
Guolin Ke's avatar
Guolin Ke committed
561
562
563
564
  inline const std::vector<std::string>& feature_names() const { return feature_names_; }

  inline void set_feature_names(const std::vector<std::string>& feature_names) {
    if (feature_names.size() != static_cast<size_t>(num_total_features_)) {
565
      Log::Fatal("Size of feature_names error, should equal with total number of features");
Guolin Ke's avatar
Guolin Ke committed
566
567
    }
    feature_names_ = std::vector<std::string>(feature_names);
Guolin Ke's avatar
Guolin Ke committed
568
    std::unordered_set<std::string> feature_name_set;
569
570
    // replace ' ' in feature_names with '_'
    bool spaceInFeatureName = false;
571
    for (auto& feature_name : feature_names_) {
572
573
      // check ascii
      if (!Common::CheckASCII(feature_name)) {
574
575
576
577
578
        Log::Fatal("Do not support non-ASCII characters in feature name.");
      }
      // check json
      if (!Common::CheckAllowedJSON(feature_name)) {
        Log::Fatal("Do not support special JSON characters in feature name.");
579
      }
580
      if (feature_name.find(' ') != std::string::npos) {
581
582
583
        spaceInFeatureName = true;
        std::replace(feature_name.begin(), feature_name.end(), ' ', '_');
      }
Guolin Ke's avatar
Guolin Ke committed
584
585
586
587
      if (feature_name_set.count(feature_name) > 0) {
        Log::Fatal("Feature (%s) appears more than one time.", feature_name.c_str());
      }
      feature_name_set.insert(feature_name);
588
    }
589
    if (spaceInFeatureName) {
590
591
      Log::Warning("Find whitespaces in feature_names, replace with underlines");
    }
Guolin Ke's avatar
Guolin Ke committed
592
  }
Guolin Ke's avatar
Guolin Ke committed
593

Guolin Ke's avatar
Guolin Ke committed
594
595
596
597
598
599
600
601
602
603
604
605
606
607
  inline std::vector<std::string> feature_infos() const {
    std::vector<std::string> bufs;
    for (int i = 0; i < num_total_features_; i++) {
      int fidx = used_feature_map_[i];
      if (fidx == -1) {
        bufs.push_back("none");
      } else {
        const auto bin_mapper = FeatureBinMapper(fidx);
        bufs.push_back(bin_mapper->bin_info());
      }
    }
    return bufs;
  }

608
609
  void ResetConfig(const char* parameters);

Guolin Ke's avatar
Guolin Ke committed
610
611
612
613
614
615
616
617
  /*! \brief Get Number of data */
  inline data_size_t num_data() const { return num_data_; }

  /*! \brief Disable copy */
  Dataset& operator=(const Dataset&) = delete;
  /*! \brief Disable copy */
  Dataset(const Dataset&) = delete;

618
  void AddFeaturesFrom(Dataset* other);
619

Nikita Titov's avatar
Nikita Titov committed
620
 private:
Guolin Ke's avatar
Guolin Ke committed
621
  std::string data_filename_;
Guolin Ke's avatar
Guolin Ke committed
622
  /*! \brief Store used features */
Guolin Ke's avatar
Guolin Ke committed
623
  std::vector<std::unique_ptr<FeatureGroup>> feature_groups_;
Guolin Ke's avatar
Guolin Ke committed
624
625
626
627
  /*! \brief Mapper from real feature index to used index*/
  std::vector<int> used_feature_map_;
  /*! \brief Number of used features*/
  int num_features_;
628
629
  /*! \brief Number of total features*/
  int num_total_features_;
Guolin Ke's avatar
Guolin Ke committed
630
631
632
633
  /*! \brief Number of total data*/
  data_size_t num_data_;
  /*! \brief Store some label level data*/
  Metadata metadata_;
Guolin Ke's avatar
Guolin Ke committed
634
635
636
637
  /*! \brief index of label column */
  int label_idx_ = 0;
  /*! \brief store feature names */
  std::vector<std::string> feature_names_;
638
639
  /*! \brief store feature names */
  static const char* binary_file_token;
Guolin Ke's avatar
Guolin Ke committed
640
641
642
643
644
645
646
  int num_groups_;
  std::vector<int> real_feature_idx_;
  std::vector<int> feature2group_;
  std::vector<int> feature2subfeature_;
  std::vector<uint64_t> group_bin_boundaries_;
  std::vector<int> group_feature_start_;
  std::vector<int> group_feature_cnt_;
Guolin Ke's avatar
Guolin Ke committed
647
  std::vector<int8_t> monotone_types_;
Guolin Ke's avatar
Guolin Ke committed
648
  std::vector<double> feature_penalty_;
Guolin Ke's avatar
Guolin Ke committed
649
  bool is_finish_load_;
650
  int max_bin_;
Belinda Trotta's avatar
Belinda Trotta committed
651
  std::vector<int32_t> max_bin_by_feature_;
652
  std::vector<std::vector<double>> forced_bin_bounds_;
653
654
655
656
  int bin_construct_sample_cnt_;
  int min_data_in_bin_;
  bool use_missing_;
  bool zero_as_missing_;
Guolin Ke's avatar
Guolin Ke committed
657
  std::vector<int> feature_need_push_zeros_;
658
659
  mutable std::vector<hist_t, Common::AlignmentAllocator<hist_t, kAlignedSize>> hist_buf_;

Guolin Ke's avatar
Guolin Ke committed
660
661
662
663
};

}  // namespace LightGBM

Guolin Ke's avatar
Guolin Ke committed
664
#endif   // LightGBM_DATA_H_