qxLib
All Classes Files Functions Variables Typedefs Enumerations Enumerator Macros
common.inl
Go to the documentation of this file.
1 /**
2 
3  @file common.inl
4  @author Khrapov
5  @date 29.04.2023
6  @copyright � Nick Khrapov, 2023. All right reserved.
7 
8 **/
9 
10 namespace qx
11 {
12 
13 template<class T>
14 constexpr T abs(T value)
15 {
16  return value < 0 ? -value : value;
17 }
18 
19 template<class T>
20 constexpr bool epsilon_equal(T left, T right, T eps)
21 {
22  return abs(left - right) < eps;
23 }
24 
25 template<class T>
26 constexpr bool epsilon_zero(T value, T eps)
27 {
28  return abs(value) < eps;
29 }
30 
31 template<class T>
32 constexpr bool epsilon_less_equal(T left, T right, T eps)
33 {
34  return left < right || epsilon_equal(left, right, eps);
35 }
36 
37 template<class T>
38 constexpr bool epsilon_greater_equal(T left, T right, T eps)
39 {
40  return left > right || epsilon_equal(left, right, eps);
41 }
42 
43 template<std::integral T>
44 constexpr bool is_odd(T val)
45 {
46  return (val & 1) == 1;
47 }
48 
49 template<std::integral T>
50 constexpr bool is_even(T val)
51 {
52  return (val & 1) == 0;
53 }
54 
55 constexpr int gcd(int nFirst, int nSecond)
56 {
57  if (nFirst == 0 || nSecond == 0)
58  return 0;
59 
60  while (nSecond != 0)
61  {
62  const int nRemainder = nFirst % nSecond;
63  nFirst = nSecond;
64  nSecond = nRemainder;
65  }
66 
67  return abs(nFirst);
68 }
69 
70 constexpr int lcm(int nFirst, int nSecond)
71 {
72  if (nFirst == 0 || nSecond == 0)
73  return 0;
74 
75  nFirst = abs(nFirst);
76  nSecond = abs(nSecond);
77 
78  return nFirst / gcd(nFirst, nSecond) * nSecond;
79 }
80 
81 template<class T>
82 inline double pow(T number, int nPower)
83 {
84  static_assert(std::is_integral_v<T> || std::is_floating_point_v<T>, "Integral or floating point required");
85 
86  const bool bNegativePower = nPower < 0;
87  const size_t nPositivePower = static_cast<size_t>(std::abs(nPower));
88 
89  double fResult = 1.0;
90  switch (nPositivePower)
91  {
92  case 0:
93  break;
94 
95  case 1:
96  fResult = static_cast<double>(number);
97  break;
98 
99  case 2:
100  fResult = static_cast<double>(number * number);
101  break;
102 
103  default:
104  const std::bitset<std::numeric_limits<int>::digits> powerBitSet(nPositivePower);
105 
106  std::array<double, std::numeric_limits<int>::digits> powers;
107 
108  powers[0] = static_cast<double>(number);
109 
110  size_t nCurPower = 1;
111  size_t nCurIndex = 1;
112 
113  while (nCurPower < nPositivePower)
114  {
115  powers[nCurIndex] = powers[nCurIndex - 1] * powers[nCurIndex - 1];
116  nCurPower *= 2;
117  nCurIndex++;
118  }
119 
120  for (size_t i = 0; i < nCurIndex; ++i)
121  if (powerBitSet.test(i))
122  fResult *= powers[i];
123 
124  break;
125  }
126 
127  return bNegativePower ? 1.0 / fResult : fResult;
128 }
129 
130 template<std::integral I>
131 I maxpot(I nValue)
132 {
133  static_assert(std::is_integral_v<I>, "Integral required");
134 
135  if (nValue == 0)
136  return 0;
137 
138  std::bitset<std::numeric_limits<I>::digits> powers(static_cast<size_t>(abs(nValue)));
139 
140  I nPow = static_cast<I>(std::numeric_limits<I>::digits - 1);
141  while (!powers.test(static_cast<size_t>(nPow)))
142  --nPow;
143 
144  return nPow;
145 }
146 
147 // trick to determine if an integer is between two integers (inclusive)
148 // with only one comparison/branch
149 // https://stackoverflow.com/a/17095534/8021662
150 QX_DISABLE_MSVC_WARNINGS(4018 4388);
151 
152 template<class T, class compare_t>
153 constexpr bool between(T left, T value, T right, compare_t compare)
154 {
155  if constexpr (std::is_enum_v<T>)
156  {
157  i64 l = static_cast<i64>(left);
158  i64 r = static_cast<i64>(right);
159  i64 v = static_cast<i64>(value);
160  return between(l, v, r, compare);
161  }
162  else if constexpr (std::is_integral_v<T> && std::is_same_v<compare_t, std::less_equal<>>)
163  {
164  return compare(static_cast<size_t>(value - left), right - left);
165  }
166  else if constexpr (std::is_floating_point_v<T> && std::is_same_v<compare_t, std::less_equal<>>)
167  {
168  return epsilon_less_equal(left, value) && epsilon_less_equal(value, right);
169  }
170  else
171  {
172  return compare(left, value) && compare(value, right);
173  }
174 }
175 
176 template<class T, class compare_t>
177 constexpr bool between(T left, T value, T right)
178 {
179  QX_PUSH_SUPPRESS_MSVC_WARNINGS(4388);
180  return between(left, value, right, compare_t());
181  QX_POP_SUPPRESS_WARNINGS();
182 }
183 
184 QX_RESTORE_MSVC_WARNINGS(4018 4388);
185 
186 } // namespace qx
constexpr int lcm(int nFirst, int nSecond)
Least common multiple.
Definition: common.inl:70
constexpr bool between(T left, T value, T right, compare_t compare)
Checks if value is between left and right.
Definition: common.inl:153
constexpr bool epsilon_greater_equal(T left, T right, T eps=std::numeric_limits< T >::epsilon())
Constexpr comparison function for a user defined epsilon values.
Definition: common.inl:38
I maxpot(I nValue)
Max power of two in integer.
Definition: common.inl:131
constexpr bool is_odd(T val)
Check if value is odd.
Definition: common.inl:44
constexpr bool epsilon_zero(T value, T eps=std::numeric_limits< T >::epsilon())
Constexpr comparison with zero for a user defined epsilon values.
Definition: common.inl:26
constexpr T abs(T value)
Constexpr absolute value.
Definition: common.inl:14
constexpr bool epsilon_less_equal(T left, T right, T eps=std::numeric_limits< T >::epsilon())
Constexpr comparison function for a user defined epsilon values.
Definition: common.inl:32
constexpr bool is_even(T val)
Check if value is even.
Definition: common.inl:50
constexpr bool epsilon_equal(T left, T right, T eps=std::numeric_limits< T >::epsilon())
Constexpr comparison function for a user defined epsilon values.
Definition: common.inl:20
double pow(T number, int nPower)
Power function for integer power.
Definition: common.inl:82
constexpr int gcd(int nFirst, int nSecond)
Greatest common divisor.
Definition: common.inl:55