📄 matrix_assign.hpp
字号:
//
// Copyright (c) 2000-2002
// Joerg Walter, Mathias Koch
//
// Permission to use, copy, modify, distribute and sell this software
// and its documentation for any purpose is hereby granted without fee,
// provided that the above copyright notice appear in all copies and
// that both that copyright notice and this permission notice appear
// in supporting documentation. The authors make no representations
// about the suitability of this software for any purpose.
// It is provided "as is" without express or implied warranty.
//
// The authors gratefully acknowledge the support of
// GeNeSys mbH & Co. KG in producing this work.
//
#ifndef BOOST_UBLAS_MATRIX_ASSIGN_H
#define BOOST_UBLAS_MATRIX_ASSIGN_H
#include <boost/numeric/ublas/config.hpp>
#include <boost/numeric/ublas/matrix_expression.hpp>
// Iterators based on ideas of Jeremy Siek
namespace boost { namespace numeric { namespace ublas {
template<class E1, class E2>
BOOST_UBLAS_INLINE
bool equals (const matrix_expression<E1> &e1, const matrix_expression<E2> &e2) {
typedef BOOST_UBLAS_TYPENAME type_traits<BOOST_UBLAS_TYPENAME promote_traits<BOOST_UBLAS_TYPENAME E1::value_type,
BOOST_UBLAS_TYPENAME E2::value_type>::promote_type>::real_type real_type;
#ifndef __GNUC__
return norm_inf (e1 - e2) < BOOST_UBLAS_TYPE_CHECK_EPSILON *
std::max<real_type> (std::max<real_type> (norm_inf (e1),
norm_inf (e2)),
BOOST_UBLAS_TYPE_CHECK_MIN);
#else
// GCC 3.1, oops?!
return norm_inf (e1 - e2) < BOOST_UBLAS_TYPE_CHECK_EPSILON *
std::max (real_type (std::max (real_type (norm_inf (e1)), real_type (norm_inf (e2)))),
real_type (BOOST_UBLAS_TYPE_CHECK_MIN));
#endif
}
// Restart for sparse (proxy) assignments
template<class E>
BOOST_UBLAS_INLINE
void restart (const matrix_expression<E> &e, typename E::size_type index1, typename E::size_type index2,
typename E::const_iterator1 &it1e, typename E::const_iterator1 &it1e_end,
typename E::const_iterator2 &it2e, typename E::const_iterator2 &it2e_end, row_major_tag) {
it1e = e ().find1 (0, index1, 0);
it1e_end = e ().find1 (0, e ().size1 (), 0);
it2e = e ().find2 (1, index1, index2);
it2e_end = e ().find2 (1, index1, e ().size2 ());
if (it2e != it2e_end && it2e.index2 () == index2)
++ it2e;
}
template<class E>
BOOST_UBLAS_INLINE
void restart (const matrix_expression<E> &e, typename E::size_type index1, typename E::size_type index2,
typename E::const_iterator2 &it2e, typename E::const_iterator2 &it2e_end,
typename E::const_iterator1 &it1e, typename E::const_iterator1 &it1e_end, column_major_tag) {
it2e = e ().find2 (0, 0, index2);
it2e_end = e ().find2 (0, 0, e ().size2 ());
it1e = e ().find1 (1, index1, index2);
it1e_end = e ().find1 (1, e ().size1 (), index2);
if (it1e != it1e_end && it1e.index1 () == index1)
++ it1e;
}
template<class E>
BOOST_UBLAS_INLINE
void restart (matrix_expression<E> &e, typename E::size_type index1, typename E::size_type index2,
typename E::iterator1 &it1e, typename E::iterator1 &it1e_end,
typename E::iterator2 &it2e, typename E::iterator2 &it2e_end, row_major_tag) {
it1e = e ().find1 (0, index1, 0);
it1e_end = e ().find1 (0, e ().size1 (), 0);
it2e = e ().find2 (1, index1, index2);
it2e_end = e ().find2 (1, index1, e ().size2 ());
if (it2e != it2e_end && it2e.index2 () == index2)
++ it2e;
}
template<class E>
BOOST_UBLAS_INLINE
void restart (matrix_expression<E> &e, typename E::size_type index1, typename E::size_type index2,
typename E::iterator2 &it2e, typename E::iterator2 &it2e_end,
typename E::iterator1 &it1e, typename E::iterator1 &it1e_end, column_major_tag) {
it2e = e ().find2 (0, 0, index2);
it2e_end = e ().find2 (0, 0, e ().size2 ());
it1e = e ().find1 (1, index1, index2);
it1e_end = e ().find1 (1, e ().size1 (), index2);
if (it1e != it1e_end && it1e.index1 () == index1)
++ it1e;
}
// Iterating row major case
template<class F, class M, class T>
// This function seems to be big. So we do not let the compiler inline it.
// BOOST_UBLAS_INLINE
void iterating_matrix_assign_scalar (F, M &m, const T &t, row_major_tag) {
typedef F functor_type;
typedef typename M::difference_type difference_type;
difference_type size1 (m.size1 ());
difference_type size2 (m.size2 ());
typename M::iterator1 it1 (m.begin1 ());
BOOST_UBLAS_CHECK (size2 == 0 || m.end1 () - it1 == size1, bad_size ());
while (-- size1 >= 0) {
#ifndef BOOST_UBLAS_NO_NESTED_CLASS_RELATION
typename M::iterator2 it2 (it1.begin ());
#else
typename M::iterator2 it2 (begin (it1, iterator1_tag ()));
#endif
BOOST_UBLAS_CHECK (it1.end () - it2 == size2, bad_size ());
difference_type temp_size2 (size2);
#ifndef BOOST_UBLAS_USE_DUFF_DEVICE
while (-- temp_size2 >= 0)
functor_type () (*it2, t), ++ it2;
#else
DD (temp_size2, 4, r, (functor_type () (*it2, t), ++ it2));
#endif
++ it1;
}
}
// Iterating column major case
template<class F, class M, class T>
// This function seems to be big. So we do not let the compiler inline it.
// BOOST_UBLAS_INLINE
void iterating_matrix_assign_scalar (F, M &m, const T &t, column_major_tag) {
typedef F functor_type;
typedef typename M::difference_type difference_type;
difference_type size2 (m.size2 ());
difference_type size1 (m.size1 ());
typename M::iterator2 it2 (m.begin2 ());
BOOST_UBLAS_CHECK (size1 == 0 || m.end2 () - it2 == size2, bad_size ());
while (-- size2 >= 0) {
#ifndef BOOST_UBLAS_NO_NESTED_CLASS_RELATION
typename M::iterator1 it1 (it2.begin ());
#else
typename M::iterator1 it1 (begin (it2, iterator2_tag ()));
#endif
BOOST_UBLAS_CHECK (it2.end () - it1 == size1, bad_size ());
difference_type temp_size1 (size1);
#ifndef BOOST_UBLAS_USE_DUFF_DEVICE
while (-- temp_size1 >= 0)
functor_type () (*it1, t), ++ it1;
#else
DD (temp_size1, 4, r, (functor_type () (*it1, t), ++ it1));
#endif
++ it2;
}
}
// Indexing row major case
template<class F, class M, class T>
// This function seems to be big. So we do not let the compiler inline it.
// BOOST_UBLAS_INLINE
void indexing_matrix_assign_scalar (F, M &m, const T &t, row_major_tag) {
typedef F functor_type;
typedef typename M::difference_type difference_type;
difference_type size1 (m.size1 ());
difference_type size2 (m.size2 ());
for (difference_type i = 0; i < size1; ++ i) {
#ifndef BOOST_UBLAS_USE_DUFF_DEVICE
for (difference_type j = 0; j < size2; ++ j)
functor_type () (m (i, j), t);
#else
difference_type j (0);
DD (size2, 4, r, (functor_type () (m (i, j), t), ++ j));
#endif
}
}
// Indexing column major case
template<class F, class M, class T>
// This function seems to be big. So we do not let the compiler inline it.
// BOOST_UBLAS_INLINE
void indexing_matrix_assign_scalar (F, M &m, const T &t, column_major_tag) {
typedef F functor_type;
typedef typename M::difference_type difference_type;
difference_type size2 (m.size2 ());
difference_type size1 (m.size1 ());
for (difference_type j = 0; j < size2; ++ j)
#ifndef BOOST_UBLAS_USE_DUFF_DEVICE
for (difference_type i = 0; i < size1; ++ i) {
functor_type () (m (i, j), t);
#else
difference_type i (0);
DD (size1, 4, r, (functor_type () (m (i, j), t), ++ i));
#endif
}
}
// Dense (proxy) case
template<class F, class M, class T, class C>
// This function seems to be big. So we do not let the compiler inline it.
// BOOST_UBLAS_INLINE
void matrix_assign_scalar (F, M &m, const T &t, dense_proxy_tag, C) {
typedef F functor_type;
typedef C orientation_category;
#ifdef BOOST_UBLAS_USE_INDEXING
indexing_matrix_assign_scalar (functor_type (), m, t, orientation_category ());
#elif BOOST_UBLAS_USE_ITERATING
iterating_matrix_assign_scalar (functor_type (), m, t, orientation_category ());
#else
typedef typename M::difference_type difference_type;
difference_type size1 (m.size1 ());
difference_type size2 (m.size2 ());
if (size1 >= BOOST_UBLAS_ITERATOR_THRESHOLD &&
size2 >= BOOST_UBLAS_ITERATOR_THRESHOLD)
iterating_matrix_assign_scalar (functor_type (), m, t, orientation_category ());
else
indexing_matrix_assign_scalar (functor_type (), m, t, orientation_category ());
#endif
}
// Packed (proxy) row major case
template<class F, class M, class T>
// This function seems to be big. So we do not let the compiler inline it.
// BOOST_UBLAS_INLINE
void matrix_assign_scalar (F, M &m, const T &t, packed_proxy_tag, row_major_tag) {
typedef F functor_type;
typedef typename M::difference_type difference_type;
typename M::iterator1 it1 (m.begin1 ());
difference_type size1 (m.end1 () - it1);
while (-- size1 >= 0) {
#ifndef BOOST_UBLAS_NO_NESTED_CLASS_RELATION
typename M::iterator2 it2 (it1.begin ());
difference_type size2 (it1.end () - it2);
#else
typename M::iterator2 it2 (begin (it1, iterator1_tag ()));
difference_type size2 (end (it1, iterator1_tag ()) - it2);
#endif
while (-- size2 >= 0)
functor_type () (*it2, t), ++ it2;
++ it1;
}
}
// Packed (proxy) column major case
template<class F, class M, class T>
// This function seems to be big. So we do not let the compiler inline it.
// BOOST_UBLAS_INLINE
void matrix_assign_scalar (F, M &m, const T &t, packed_proxy_tag, column_major_tag) {
typedef F functor_type;
typedef typename M::difference_type difference_type;
typename M::iterator2 it2 (m.begin2 ());
difference_type size2 (m.end2 () - it2);
while (-- size2 >= 0) {
#ifndef BOOST_UBLAS_NO_NESTED_CLASS_RELATION
typename M::iterator1 it1 (it2.begin ());
difference_type size1 (it2.end () - it1);
#else
typename M::iterator1 it1 (begin (it2, iterator2_tag ()));
difference_type size1 (end (it2, iterator2_tag ()) - it1);
#endif
while (-- size1 >= 0)
functor_type () (*it1, t), ++ it1;
++ it2;
}
}
// Sparse (proxy) row major case
template<class F, class M, class T>
// This function seems to be big. So we do not let the compiler inline it.
// BOOST_UBLAS_INLINE
void matrix_assign_scalar (F, M &m, const T &t, sparse_proxy_tag, row_major_tag) {
typedef F functor_type;
typename M::iterator1 it1 (m.begin1 ());
typename M::iterator1 it1_end (m.end1 ());
while (it1 != it1_end) {
#ifndef BOOST_UBLAS_NO_NESTED_CLASS_RELATION
typename M::iterator2 it2 (it1.begin ());
typename M::iterator2 it2_end (it1.end ());
#else
typename M::iterator2 it2 (begin (it1, iterator1_tag ()));
typename M::iterator2 it2_end (end (it1, iterator1_tag ()));
#endif
while (it2 != it2_end)
functor_type () (*it2, t), ++ it2;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -