utility.hpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337
  1. /* Boost interval/utility.hpp template implementation file
  2. *
  3. * Copyright 2000 Jens Maurer
  4. * Copyright 2002-2003 Hervé Brönnimann, Guillaume Melquiond, Sylvain Pion
  5. *
  6. * Distributed under the Boost Software License, Version 1.0.
  7. * (See accompanying file LICENSE_1_0.txt or
  8. * copy at http://www.boost.org/LICENSE_1_0.txt)
  9. */
  10. #ifndef BOOST_NUMERIC_INTERVAL_UTILITY_HPP
  11. #define BOOST_NUMERIC_INTERVAL_UTILITY_HPP
  12. #include <boost/config.hpp>
  13. #include <boost/numeric/interval/detail/interval_prototype.hpp>
  14. #include <boost/numeric/interval/detail/test_input.hpp>
  15. #include <boost/numeric/interval/detail/bugs.hpp>
  16. #include <algorithm>
  17. #include <utility>
  18. /*
  19. * Implementation of simple functions
  20. */
  21. namespace boost {
  22. namespace numeric {
  23. /*
  24. * Utility Functions
  25. */
  26. template<class T, class Policies> inline
  27. const T& lower(const interval<T, Policies>& x)
  28. {
  29. return x.lower();
  30. }
  31. template<class T, class Policies> inline
  32. const T& upper(const interval<T, Policies>& x)
  33. {
  34. return x.upper();
  35. }
  36. template<class T, class Policies> inline
  37. T checked_lower(const interval<T, Policies>& x)
  38. {
  39. if (empty(x)) {
  40. typedef typename Policies::checking checking;
  41. return checking::nan();
  42. }
  43. return x.lower();
  44. }
  45. template<class T, class Policies> inline
  46. T checked_upper(const interval<T, Policies>& x)
  47. {
  48. if (empty(x)) {
  49. typedef typename Policies::checking checking;
  50. return checking::nan();
  51. }
  52. return x.upper();
  53. }
  54. template<class T, class Policies> inline
  55. T width(const interval<T, Policies>& x)
  56. {
  57. if (interval_lib::detail::test_input(x)) return static_cast<T>(0);
  58. typename Policies::rounding rnd;
  59. return rnd.sub_up(x.upper(), x.lower());
  60. }
  61. template<class T, class Policies> inline
  62. T median(const interval<T, Policies>& x)
  63. {
  64. if (interval_lib::detail::test_input(x)) {
  65. typedef typename Policies::checking checking;
  66. return checking::nan();
  67. }
  68. typename Policies::rounding rnd;
  69. return rnd.median(x.lower(), x.upper());
  70. }
  71. template<class T, class Policies> inline
  72. interval<T, Policies> widen(const interval<T, Policies>& x, const T& v)
  73. {
  74. if (interval_lib::detail::test_input(x))
  75. return interval<T, Policies>::empty();
  76. typename Policies::rounding rnd;
  77. return interval<T, Policies>(rnd.sub_down(x.lower(), v),
  78. rnd.add_up (x.upper(), v), true);
  79. }
  80. /*
  81. * Set-like operations
  82. */
  83. template<class T, class Policies> inline
  84. bool empty(const interval<T, Policies>& x)
  85. {
  86. return interval_lib::detail::test_input(x);
  87. }
  88. template<class T, class Policies> inline
  89. bool zero_in(const interval<T, Policies>& x)
  90. {
  91. if (interval_lib::detail::test_input(x)) return false;
  92. return (!interval_lib::user::is_pos(x.lower())) &&
  93. (!interval_lib::user::is_neg(x.upper()));
  94. }
  95. template<class T, class Policies> inline
  96. bool in_zero(const interval<T, Policies>& x) // DEPRECATED
  97. {
  98. return zero_in<T, Policies>(x);
  99. }
  100. template<class T, class Policies> inline
  101. bool in(const T& x, const interval<T, Policies>& y)
  102. {
  103. if (interval_lib::detail::test_input(x, y)) return false;
  104. return y.lower() <= x && x <= y.upper();
  105. }
  106. template<class T, class Policies> inline
  107. bool subset(const interval<T, Policies>& x,
  108. const interval<T, Policies>& y)
  109. {
  110. if (empty(x)) return true;
  111. return !empty(y) && y.lower() <= x.lower() && x.upper() <= y.upper();
  112. }
  113. template<class T, class Policies1, class Policies2> inline
  114. bool proper_subset(const interval<T, Policies1>& x,
  115. const interval<T, Policies2>& y)
  116. {
  117. if (empty(y)) return false;
  118. if (empty(x)) return true;
  119. return y.lower() <= x.lower() && x.upper() <= y.upper() &&
  120. (y.lower() != x.lower() || x.upper() != y.upper());
  121. }
  122. template<class T, class Policies1, class Policies2> inline
  123. bool overlap(const interval<T, Policies1>& x,
  124. const interval<T, Policies2>& y)
  125. {
  126. if (interval_lib::detail::test_input(x, y)) return false;
  127. return (x.lower() <= y.lower() && y.lower() <= x.upper()) ||
  128. (y.lower() <= x.lower() && x.lower() <= y.upper());
  129. }
  130. template<class T, class Policies> inline
  131. bool singleton(const interval<T, Policies>& x)
  132. {
  133. return !empty(x) && x.lower() == x.upper();
  134. }
  135. template<class T, class Policies1, class Policies2> inline
  136. bool equal(const interval<T, Policies1>& x, const interval<T, Policies2>& y)
  137. {
  138. if (empty(x)) return empty(y);
  139. return !empty(y) && x.lower() == y.lower() && x.upper() == y.upper();
  140. }
  141. template<class T, class Policies> inline
  142. interval<T, Policies> intersect(const interval<T, Policies>& x,
  143. const interval<T, Policies>& y)
  144. {
  145. BOOST_USING_STD_MIN();
  146. BOOST_USING_STD_MAX();
  147. if (interval_lib::detail::test_input(x, y))
  148. return interval<T, Policies>::empty();
  149. const T& l = max BOOST_PREVENT_MACRO_SUBSTITUTION(x.lower(), y.lower());
  150. const T& u = min BOOST_PREVENT_MACRO_SUBSTITUTION(x.upper(), y.upper());
  151. if (l <= u) return interval<T, Policies>(l, u, true);
  152. else return interval<T, Policies>::empty();
  153. }
  154. template<class T, class Policies> inline
  155. interval<T, Policies> hull(const interval<T, Policies>& x,
  156. const interval<T, Policies>& y)
  157. {
  158. BOOST_USING_STD_MIN();
  159. BOOST_USING_STD_MAX();
  160. bool bad_x = interval_lib::detail::test_input(x);
  161. bool bad_y = interval_lib::detail::test_input(y);
  162. if (bad_x)
  163. if (bad_y) return interval<T, Policies>::empty();
  164. else return y;
  165. else
  166. if (bad_y) return x;
  167. return interval<T, Policies>(min BOOST_PREVENT_MACRO_SUBSTITUTION(x.lower(), y.lower()),
  168. max BOOST_PREVENT_MACRO_SUBSTITUTION(x.upper(), y.upper()), true);
  169. }
  170. template<class T, class Policies> inline
  171. interval<T, Policies> hull(const interval<T, Policies>& x, const T& y)
  172. {
  173. BOOST_USING_STD_MIN();
  174. BOOST_USING_STD_MAX();
  175. bool bad_x = interval_lib::detail::test_input(x);
  176. bool bad_y = interval_lib::detail::test_input<T, Policies>(y);
  177. if (bad_y)
  178. if (bad_x) return interval<T, Policies>::empty();
  179. else return x;
  180. else
  181. if (bad_x) return interval<T, Policies>(y, y, true);
  182. return interval<T, Policies>(min BOOST_PREVENT_MACRO_SUBSTITUTION(x.lower(), y),
  183. max BOOST_PREVENT_MACRO_SUBSTITUTION(x.upper(), y), true);
  184. }
  185. template<class T, class Policies> inline
  186. interval<T, Policies> hull(const T& x, const interval<T, Policies>& y)
  187. {
  188. BOOST_USING_STD_MIN();
  189. BOOST_USING_STD_MAX();
  190. bool bad_x = interval_lib::detail::test_input<T, Policies>(x);
  191. bool bad_y = interval_lib::detail::test_input(y);
  192. if (bad_x)
  193. if (bad_y) return interval<T, Policies>::empty();
  194. else return y;
  195. else
  196. if (bad_y) return interval<T, Policies>(x, x, true);
  197. return interval<T, Policies>(min BOOST_PREVENT_MACRO_SUBSTITUTION(x, y.lower()),
  198. max BOOST_PREVENT_MACRO_SUBSTITUTION(x, y.upper()), true);
  199. }
  200. template<class T> inline
  201. interval<T> hull(const T& x, const T& y)
  202. {
  203. return interval<T>::hull(x, y);
  204. }
  205. template<class T, class Policies> inline
  206. std::pair<interval<T, Policies>, interval<T, Policies> >
  207. bisect(const interval<T, Policies>& x)
  208. {
  209. typedef interval<T, Policies> I;
  210. if (interval_lib::detail::test_input(x))
  211. return std::pair<I,I>(I::empty(), I::empty());
  212. const T m = median(x);
  213. return std::pair<I,I>(I(x.lower(), m, true), I(m, x.upper(), true));
  214. }
  215. /*
  216. * Elementary functions
  217. */
  218. template<class T, class Policies> inline
  219. T norm(const interval<T, Policies>& x)
  220. {
  221. typedef interval<T, Policies> I;
  222. if (interval_lib::detail::test_input(x)) {
  223. typedef typename Policies::checking checking;
  224. return checking::nan();
  225. }
  226. BOOST_USING_STD_MAX();
  227. return max BOOST_PREVENT_MACRO_SUBSTITUTION(static_cast<T>(-x.lower()), x.upper());
  228. }
  229. template<class T, class Policies> inline
  230. interval<T, Policies> abs(const interval<T, Policies>& x)
  231. {
  232. typedef interval<T, Policies> I;
  233. if (interval_lib::detail::test_input(x))
  234. return I::empty();
  235. if (!interval_lib::user::is_neg(x.lower())) return x;
  236. if (!interval_lib::user::is_pos(x.upper())) return -x;
  237. BOOST_USING_STD_MAX();
  238. return I(static_cast<T>(0), max BOOST_PREVENT_MACRO_SUBSTITUTION(static_cast<T>(-x.lower()), x.upper()), true);
  239. }
  240. template<class T, class Policies> inline
  241. interval<T, Policies> max BOOST_PREVENT_MACRO_SUBSTITUTION (const interval<T, Policies>& x,
  242. const interval<T, Policies>& y)
  243. {
  244. typedef interval<T, Policies> I;
  245. if (interval_lib::detail::test_input(x, y))
  246. return I::empty();
  247. BOOST_USING_STD_MAX();
  248. return I(max BOOST_PREVENT_MACRO_SUBSTITUTION(x.lower(), y.lower()), max BOOST_PREVENT_MACRO_SUBSTITUTION(x.upper(), y.upper()), true);
  249. }
  250. template<class T, class Policies> inline
  251. interval<T, Policies> max BOOST_PREVENT_MACRO_SUBSTITUTION (const interval<T, Policies>& x, const T& y)
  252. {
  253. typedef interval<T, Policies> I;
  254. if (interval_lib::detail::test_input(x, y))
  255. return I::empty();
  256. BOOST_USING_STD_MAX();
  257. return I(max BOOST_PREVENT_MACRO_SUBSTITUTION(x.lower(), y), max BOOST_PREVENT_MACRO_SUBSTITUTION(x.upper(), y), true);
  258. }
  259. template<class T, class Policies> inline
  260. interval<T, Policies> max BOOST_PREVENT_MACRO_SUBSTITUTION (const T& x, const interval<T, Policies>& y)
  261. {
  262. typedef interval<T, Policies> I;
  263. if (interval_lib::detail::test_input(x, y))
  264. return I::empty();
  265. BOOST_USING_STD_MAX();
  266. return I(max BOOST_PREVENT_MACRO_SUBSTITUTION(x, y.lower()), max BOOST_PREVENT_MACRO_SUBSTITUTION(x, y.upper()), true);
  267. }
  268. template<class T, class Policies> inline
  269. interval<T, Policies> min BOOST_PREVENT_MACRO_SUBSTITUTION (const interval<T, Policies>& x,
  270. const interval<T, Policies>& y)
  271. {
  272. typedef interval<T, Policies> I;
  273. if (interval_lib::detail::test_input(x, y))
  274. return I::empty();
  275. BOOST_USING_STD_MIN();
  276. return I(min BOOST_PREVENT_MACRO_SUBSTITUTION(x.lower(), y.lower()), min BOOST_PREVENT_MACRO_SUBSTITUTION(x.upper(), y.upper()), true);
  277. }
  278. template<class T, class Policies> inline
  279. interval<T, Policies> min BOOST_PREVENT_MACRO_SUBSTITUTION (const interval<T, Policies>& x, const T& y)
  280. {
  281. typedef interval<T, Policies> I;
  282. if (interval_lib::detail::test_input(x, y))
  283. return I::empty();
  284. BOOST_USING_STD_MIN();
  285. return I(min BOOST_PREVENT_MACRO_SUBSTITUTION(x.lower(), y), min BOOST_PREVENT_MACRO_SUBSTITUTION(x.upper(), y), true);
  286. }
  287. template<class T, class Policies> inline
  288. interval<T, Policies> min BOOST_PREVENT_MACRO_SUBSTITUTION (const T& x, const interval<T, Policies>& y)
  289. {
  290. typedef interval<T, Policies> I;
  291. if (interval_lib::detail::test_input(x, y))
  292. return I::empty();
  293. BOOST_USING_STD_MIN();
  294. return I(min BOOST_PREVENT_MACRO_SUBSTITUTION(x, y.lower()), min BOOST_PREVENT_MACRO_SUBSTITUTION(x, y.upper()), true);
  295. }
  296. } // namespace numeric
  297. } // namespace boost
  298. #endif // BOOST_NUMERIC_INTERVAL_UTILITY_HPP