• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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