lower_bound.h 3.5 KB
Newer Older
羽飞's avatar
羽飞 已提交
1
/* Copyright (c) 2021 OceanBase and/or its affiliates. All rights reserved.
羽飞's avatar
羽飞 已提交
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115
miniob is licensed under Mulan PSL v2.
You can use this software according to the terms and conditions of the Mulan PSL v2.
You may obtain a copy of Mulan PSL v2 at:
         http://license.coscl.org.cn/MulanPSL2
THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
See the Mulan PSL v2 for more details. */

//
//
// Created by Wangyunlai
//

#pragma once

#include <iterator>

namespace common {
/**
 * @brief 找到不小于指定值的元素
 * @note  数组中的元素都不相等
 * @param first 已经排序的序列中第一个元素
 * @param last  已经排序的序列中的最后一个元素的后一个
 * @param val   要查找的元素的值
 * @param comp  比较函数。相等返回0,小于返回负数,大于返回正数
 * @param _found 如果给定,会返回是否存在于当前的序列中
 * @return ForwardIterator 指向lower bound结果的iterator。如果大于最大的值,那么会指向last
 */
template <typename ForwardIterator, typename T, typename Compare>
ForwardIterator lower_bound(ForwardIterator first, ForwardIterator last,
			    const T &val, Compare comp, bool *_found = nullptr)
{
  bool found = false;
  ForwardIterator iter;
  const auto count = std::distance(first, last);
  auto last_count = count;
  while (last_count > 0) {
    iter = first;
    auto step = last_count / 2;
    std::advance(iter, step);
    int result = comp(*iter, val);
    if (0 == result) {
      first = iter;
      found = true;
      break;
    }
    if (result < 0) {
      first = ++iter;
      last_count -= step + 1;
    } else {
      last_count = step;
    }
  }

  if (_found) {
    *_found = found;
  }
  return first;
}

template <typename T>
class Comparator
{
public:
  int operator() (const T &v1, const T &v2) const
  {
    if (v1 < v2) {
      return -1;
    }
    if (v1 > v2) {
      return 1;
    }
    return 0;
  }
};

template <typename ForwardIterator, typename T>
ForwardIterator lower_bound(ForwardIterator first, ForwardIterator last, const T &val, bool *_found = nullptr)
{
  return lower_bound<ForwardIterator, T, Comparator<T>>(first, last, val, Comparator<T>(), _found);
}

template <typename T, typename Distance = ptrdiff_t>
class BinaryIterator : public std::iterator<std::random_access_iterator_tag, T *, Distance>
{
public: 
  BinaryIterator() = default;
  BinaryIterator(size_t item_num, T *data) : item_num_(item_num), data_(data)
  {}

  BinaryIterator &operator+= (int n) { data_ += (item_num_ * n); return *this; }
  BinaryIterator &operator-= (int n) { return this->operator+ (-n); }
  BinaryIterator &operator++() { return this->operator +=(1); }
  BinaryIterator operator++(int) { BinaryIterator tmp(*this); this->operator++(); return tmp; }
  BinaryIterator &operator--() { return this->operator += (-1); }
  BinaryIterator operator--(int) { BinaryIterator tmp(*this); this->operator += (-1); return tmp; }

  bool operator == (const BinaryIterator &other) const { return data_ == other.data_; }
  bool operator != (const BinaryIterator &other) const { return ! (this->operator ==(other)); }

  T * operator *() { return data_; }
  T * operator ->() { return data_; }

  friend Distance operator - (const BinaryIterator &left, const BinaryIterator &right)
  {
    return (left.data_ - right.data_) / left.item_num_;
  }

private:
  size_t item_num_ = 0;
  T * data_ = nullptr;
};
} // namespace common