3
0
mirror of https://github.com/triqs/dft_tools synced 2025-01-12 05:58:18 +01:00

arrays: fix fold and update documentation

- fold was not correct in e.g. passing an int as init instead of a
  double (was leading to narrowing in return).
- better return type deduction.
- there was an error in the doc (order of argument in the lambda !)
- add a more complex example (Frobenius norm of matrices).
This commit is contained in:
Olivier Parcollet 2014-04-09 21:24:03 +02:00
parent db59c8e5fc
commit a730f093d6
2 changed files with 64 additions and 23 deletions

View File

@ -70,7 +70,7 @@ fold
If `f` is a function, or a function object of synopsis (T, R being 2 types) ::
R f ( T, R )
R f (R , T)
then ::
@ -87,7 +87,7 @@ fold
then ::
fold (f) ( A, R init = R() ) = f( f( f( ... f( a(0,1), f(a(0,0), init)))))
fold (f) ( A, R init = R() ) = f(f(f(f(init, a(0,0)), a(0,1)),a(0,2)),a(0,3), ....)
Note that :
@ -102,9 +102,39 @@ fold
Many algorithms can be written in form of map/fold.
The function :ref:`arr_fnt_sum` which returns the sum of all the elements of the array is implemented as ::
template <class A>
typename A::value_type sum(A const & a) { return fold ( std::plus<typename A::value_type>()) (a); }
template <class A>
typename A::value_type sum(A const & a) { return fold ( std::plus<>()) (a); }
or the Frobenius norm of a matrix,
.. math::
\sum_{i=0}^{N-1} \sum_{j=0}^{N-1} | a_{ij} | ^2
reads :
.. compileblock::
#include <triqs/arrays.hpp>
#include <triqs/arrays/functional/fold.hpp>
using namespace triqs;
double frobenius_norm (arrays::matrix<double> const& a) {
auto l= [](double r, double x) {
auto ab = std::abs(x);
return r + ab * ab;
};
return std::sqrt(arrays::fold(l)(a,0));
}
int main() {
// declare and init a matrix
clef::placeholder<0> i_; clef::placeholder<1> j_;
arrays::matrix<double> A (2,2); A(i_,j_) << i_ + j_/2.0;
std::cout<< "A = " << A << std::endl;
std::cout<< "||A|| = " << frobenius_norm(A) << std::endl;
}
Note in this example :
@ -112,7 +142,7 @@ fold
* the simplicity of the code
* the genericity : it is valid for any dimension of array.
* internally, the library will rewrite it as a series of for loop, ordered in the TraversalOrder of the array
and inline the plus operator.
and inline the lambda.

View File

@ -25,29 +25,40 @@
#include <boost/function.hpp>
#include "../array.hpp"
namespace triqs { namespace arrays {
namespace triqs {
namespace arrays {
template<class F>
struct fold_worker {
F f;
template<class A, class R> struct fold_func_adaptor {
F const & f; A const & a; R & r;
template<typename ... Args> void operator()(Args const & ... args) { r = f(r,a(args...)); }
};
template <class F> struct fold_worker {
F f;
template<class A, class R>
R operator() (A const & a, R init) const {
foreach(a, fold_func_adaptor<A,R> {f,a,init});
return init;
}
template<class A> typename A::value_type operator() (A const & a) const { return (*this)(a, typename A::value_type{});}
template <class A, class R> struct fold_func_adaptor {
F const& f;
A const& a;
R& r;
template <typename... Args> void operator()(Args const&... args) { r = f(r, a(args...)); }
};
template<class F> fold_worker<F> fold (F f) { return {std::move(f)};}
template <class A, class R>
auto operator()(A const& a,
R init) const -> typename std::decay<typename std::result_of<F(typename A::value_type, R)>::type>::type {
// to take into account that f may be double,double -> double, while one passes 0 (an int...)
// R = int, R2= double in such case, and the result will be a double, or narrowing will occur
using R2 = typename std::decay<typename std::result_of<F(typename A::value_type, R)>::type>::type;
R2 r2 = init;
foreach(a, fold_func_adaptor<A, R2>{f, a, r2});
return r2;
}
}}//namespace triqs::arrays
template <class A> typename A::value_type operator()(A const& a) const {
return (*this)(a, typename A::value_type{});
}
};
template <class F> fold_worker<F> fold(F f) {
return {std::move(f)};
}
}
} // namespace triqs::arrays
#endif