1 /* Boost example/newton-raphson.cpp
2 * Newton iteration for intervals
3 *
4 * Copyright 2003 Guillaume Melquiond
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
11 #include <boost/numeric/interval.hpp>
12 #include <vector>
13 #include <algorithm>
14 #include <utility>
15 #include <iostream>
16 #include <iomanip>
17
f(const I & x)18 template <class I> I f(const I& x)
19 { return x * (x - 1.) * (x - 2.) * (x - 3.) * (x - 4.); }
f_diff(const I & x)20 template <class I> I f_diff(const I& x)
21 { return (((5. * x - 40.) * x + 105.) * x - 100.) * x + 24.; }
22
23 static const double max_width = 1e-10;
24 static const double alpha = 0.75;
25
26 using namespace boost;
27 using namespace numeric;
28 using namespace interval_lib;
29
30 // First method: no empty intervals
31
32 typedef interval<double> I1_aux;
33 typedef unprotect<I1_aux>::type I1;
34
newton_raphson(const I1 & xs)35 std::vector<I1> newton_raphson(const I1& xs) {
36 std::vector<I1> l, res;
37 I1 vf, vd, x, x1, x2;
38 l.push_back(xs);
39 while (!l.empty()) {
40 x = l.back();
41 l.pop_back();
42 bool x2_used;
43 double xx = median(x);
44 vf = f<I1>(xx);
45 vd = f_diff<I1>(x);
46 if (zero_in(vf) && zero_in(vd)) {
47 x1 = I1::whole();
48 x2_used = false;
49 } else {
50 x1 = xx - division_part1(vf, vd, x2_used);
51 if (x2_used) x2 = xx - division_part2(vf, vd);
52 }
53 if (overlap(x1, x)) x1 = intersect(x, x1);
54 else if (x2_used) { x1 = x2; x2_used = false; }
55 else continue;
56 if (x2_used) {
57 if (overlap(x2, x)) x2 = intersect(x, x2);
58 else x2_used = false;
59 }
60 if (x2_used && width(x2) > width(x1)) std::swap(x1, x2);
61 if (!zero_in(f(x1))) {
62 if (x2_used) { x1 = x2; x2_used = false; }
63 else continue;
64 }
65 if (width(x1) < max_width) res.push_back(x1);
66 else if (width(x1) > alpha * width(x)) {
67 std::pair<I1, I1> p = bisect(x);
68 if (zero_in(f(p.first))) l.push_back(p.first);
69 x2 = p.second;
70 x2_used = true;
71 } else l.push_back(x1);
72 if (x2_used && zero_in(f(x2))) {
73 if (width(x2) < max_width) res.push_back(x2);
74 else l.push_back(x2);
75 }
76 }
77 return res;
78 }
79
80 // Second method: with empty intervals
81
82 typedef change_checking<I1_aux, checking_no_nan<double> >::type I2_aux;
83 typedef unprotect<I2_aux>::type I2;
84
newton_raphson(const I2 & xs)85 std::vector<I2> newton_raphson(const I2& xs) {
86 std::vector<I2> l, res;
87 I2 vf, vd, x, x1, x2;
88 l.push_back(xs);
89 while (!l.empty()) {
90 x = l.back();
91 l.pop_back();
92 double xx = median(x);
93 vf = f<I2>(xx);
94 vd = f_diff<I2>(x);
95 if (zero_in(vf) && zero_in(vd)) {
96 x1 = x;
97 x2 = I2::empty();
98 } else {
99 bool x2_used;
100 x1 = intersect(x, xx - division_part1(vf, vd, x2_used));
101 x2 = intersect(x, xx - division_part2(vf, vd, x2_used));
102 }
103 if (width(x2) > width(x1)) std::swap(x1, x2);
104 if (empty(x1) || !zero_in(f(x1))) {
105 if (!empty(x2)) { x1 = x2; x2 = I2::empty(); }
106 else continue;
107 }
108 if (width(x1) < max_width) res.push_back(x1);
109 else if (width(x1) > alpha * width(x)) {
110 std::pair<I2, I2> p = bisect(x);
111 if (zero_in(f(p.first))) l.push_back(p.first);
112 x2 = p.second;
113 } else l.push_back(x1);
114 if (!empty(x2) && zero_in(f(x2))) {
115 if (width(x2) < max_width) res.push_back(x2);
116 else l.push_back(x2);
117 }
118 }
119 return res;
120 }
121
122 template<class T, class Policies>
operator <<(std::ostream & os,const boost::numeric::interval<T,Policies> & x)123 std::ostream &operator<<(std::ostream &os,
124 const boost::numeric::interval<T, Policies> &x) {
125 os << "[" << x.lower() << ", " << x.upper() << "]";
126 return os;
127 }
128
main()129 int main() {
130 {
131 I1_aux::traits_type::rounding rnd;
132 std::vector<I1> res = newton_raphson(I1(-1, 5.1));
133 std::cout << "Results: " << std::endl << std::setprecision(12);
134 for(std::vector<I1>::const_iterator i = res.begin(); i != res.end(); ++i)
135 std::cout << " " << *i << std::endl;
136 std::cout << std::endl;
137 }
138 {
139 I2_aux::traits_type::rounding rnd;
140 std::vector<I2> res = newton_raphson(I2(-1, 5.1));
141 std::cout << "Results: " << std::endl << std::setprecision(12);
142 for(std::vector<I2>::const_iterator i = res.begin(); i != res.end(); ++i)
143 std::cout << " " << *i << std::endl;
144 std::cout << std::endl;
145 }
146 }
147