SparseMatrix.h
Go to the documentation of this file.
00001 // This file is part of Eigen, a lightweight C++ template library
00002 // for linear algebra.
00003 //
00004 // Copyright (C) 2008-2010 Gael Guennebaud <gael.guennebaud@inria.fr>
00005 //
00006 // Eigen is free software; you can redistribute it and/or
00007 // modify it under the terms of the GNU Lesser General Public
00008 // License as published by the Free Software Foundation; either
00009 // version 3 of the License, or (at your option) any later version.
00010 //
00011 // Alternatively, you can redistribute it and/or
00012 // modify it under the terms of the GNU General Public License as
00013 // published by the Free Software Foundation; either version 2 of
00014 // the License, or (at your option) any later version.
00015 //
00016 // Eigen is distributed in the hope that it will be useful, but WITHOUT ANY
00017 // WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
00018 // FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License or the
00019 // GNU General Public License for more details.
00020 //
00021 // You should have received a copy of the GNU Lesser General Public
00022 // License and a copy of the GNU General Public License along with
00023 // Eigen. If not, see <http://www.gnu.org/licenses/>.
00024 
00025 #ifndef EIGEN_SPARSEMATRIX_H
00026 #define EIGEN_SPARSEMATRIX_H
00027 
00048 namespace internal {
00049 template<typename _Scalar, int _Options, typename _Index>
00050 struct traits<SparseMatrix<_Scalar, _Options, _Index> >
00051 {
00052   typedef _Scalar Scalar;
00053   typedef _Index Index;
00054   typedef Sparse StorageKind;
00055   typedef MatrixXpr XprKind;
00056   enum {
00057     RowsAtCompileTime = Dynamic,
00058     ColsAtCompileTime = Dynamic,
00059     MaxRowsAtCompileTime = Dynamic,
00060     MaxColsAtCompileTime = Dynamic,
00061     Flags = _Options | NestByRefBit | LvalueBit,
00062     CoeffReadCost = NumTraits<Scalar>::ReadCost,
00063     SupportedAccessPatterns = InnerRandomAccessPattern
00064   };
00065 };
00066 
00067 } // end namespace internal
00068 
00069 template<typename _Scalar, int _Options, typename _Index>
00070 class SparseMatrix
00071   : public SparseMatrixBase<SparseMatrix<_Scalar, _Options, _Index> >
00072 {
00073   public:
00074     EIGEN_SPARSE_PUBLIC_INTERFACE(SparseMatrix)
00075 //     using Base::operator=;
00076     EIGEN_SPARSE_INHERIT_ASSIGNMENT_OPERATOR(SparseMatrix, +=)
00077     EIGEN_SPARSE_INHERIT_ASSIGNMENT_OPERATOR(SparseMatrix, -=)
00078     // FIXME: why are these operator already alvailable ???
00079     // EIGEN_SPARSE_INHERIT_SCALAR_ASSIGNMENT_OPERATOR(SparseMatrix, *=)
00080     // EIGEN_SPARSE_INHERIT_SCALAR_ASSIGNMENT_OPERATOR(SparseMatrix, /=)
00081 
00082     typedef MappedSparseMatrix<Scalar,Flags> Map;
00083     using Base::IsRowMajor;
00084     typedef CompressedStorage<Scalar,Index> Storage;
00085     enum {
00086       Options = _Options
00087     };
00088 
00089   protected:
00090 
00091     typedef SparseMatrix<Scalar,(Flags&~RowMajorBit)|(IsRowMajor?RowMajorBit:0)> TransposedSparseMatrix;
00092 
00093     Index m_outerSize;
00094     Index m_innerSize;
00095     Index* m_outerIndex;
00096     CompressedStorage<Scalar,Index> m_data;
00097 
00098   public:
00099 
00100     inline Index rows() const { return IsRowMajor ? m_outerSize : m_innerSize; }
00101     inline Index cols() const { return IsRowMajor ? m_innerSize : m_outerSize; }
00102 
00103     inline Index innerSize() const { return m_innerSize; }
00104     inline Index outerSize() const { return m_outerSize; }
00105     inline Index innerNonZeros(Index j) const { return m_outerIndex[j+1]-m_outerIndex[j]; }
00106 
00107     inline const Scalar* _valuePtr() const { return &m_data.value(0); }
00108     inline Scalar* _valuePtr() { return &m_data.value(0); }
00109 
00110     inline const Index* _innerIndexPtr() const { return &m_data.index(0); }
00111     inline Index* _innerIndexPtr() { return &m_data.index(0); }
00112 
00113     inline const Index* _outerIndexPtr() const { return m_outerIndex; }
00114     inline Index* _outerIndexPtr() { return m_outerIndex; }
00115 
00116     inline Storage& data() { return m_data; }
00117     inline const Storage& data() const { return m_data; }
00118 
00119     inline Scalar coeff(Index row, Index col) const
00120     {
00121       const Index outer = IsRowMajor ? row : col;
00122       const Index inner = IsRowMajor ? col : row;
00123       return m_data.atInRange(m_outerIndex[outer], m_outerIndex[outer+1], inner);
00124     }
00125 
00126     inline Scalar& coeffRef(Index row, Index col)
00127     {
00128       const Index outer = IsRowMajor ? row : col;
00129       const Index inner = IsRowMajor ? col : row;
00130 
00131       Index start = m_outerIndex[outer];
00132       Index end = m_outerIndex[outer+1];
00133       eigen_assert(end>=start && "you probably called coeffRef on a non finalized matrix");
00134       eigen_assert(end>start && "coeffRef cannot be called on a zero coefficient");
00135       const Index p = m_data.searchLowerIndex(start,end-1,inner);
00136       eigen_assert((p<end) && (m_data.index(p)==inner) && "coeffRef cannot be called on a zero coefficient");
00137       return m_data.value(p);
00138     }
00139 
00140   public:
00141 
00142     class InnerIterator;
00143 
00145     inline void setZero()
00146     {
00147       m_data.clear();
00148       memset(m_outerIndex, 0, (m_outerSize+1)*sizeof(Index));
00149     }
00150 
00152     inline Index nonZeros() const  { return static_cast<Index>(m_data.size()); }
00153 
00155     inline void reserve(Index reserveSize)
00156     {
00157       m_data.reserve(reserveSize);
00158     }
00159 
00160     //--- low level purely coherent filling ---
00161 
00171     inline Scalar& insertBack(Index row, Index col)
00172     {
00173       return insertBackByOuterInner(IsRowMajor?row:col, IsRowMajor?col:row);
00174     }
00175 
00177     inline Scalar& insertBackByOuterInner(Index outer, Index inner)
00178     {
00179       eigen_assert(size_t(m_outerIndex[outer+1]) == m_data.size() && "Invalid ordered insertion (invalid outer index)");
00180       eigen_assert( (m_outerIndex[outer+1]-m_outerIndex[outer]==0 || m_data.index(m_data.size()-1)<inner) && "Invalid ordered insertion (invalid inner index)");
00181       Index p = m_outerIndex[outer+1];
00182       ++m_outerIndex[outer+1];
00183       m_data.append(0, inner);
00184       return m_data.value(p);
00185     }
00186 
00188     inline Scalar& insertBackByOuterInnerUnordered(Index outer, Index inner)
00189     {
00190       Index p = m_outerIndex[outer+1];
00191       ++m_outerIndex[outer+1];
00192       m_data.append(0, inner);
00193       return m_data.value(p);
00194     }
00195 
00197     inline void startVec(Index outer)
00198     {
00199       eigen_assert(m_outerIndex[outer]==int(m_data.size()) && "You must call startVec for each inner vector sequentially");
00200       eigen_assert(m_outerIndex[outer+1]==0 && "You must call startVec for each inner vector sequentially");
00201       m_outerIndex[outer+1] = m_outerIndex[outer];
00202     }
00203 
00204     //---
00205 
00214     EIGEN_DONT_INLINE Scalar& insert(Index row, Index col)
00215     {
00216       const Index outer = IsRowMajor ? row : col;
00217       const Index inner = IsRowMajor ? col : row;
00218 
00219       Index previousOuter = outer;
00220       if (m_outerIndex[outer+1]==0)
00221       {
00222         // we start a new inner vector
00223         while (previousOuter>=0 && m_outerIndex[previousOuter]==0)
00224         {
00225           m_outerIndex[previousOuter] = static_cast<Index>(m_data.size());
00226           --previousOuter;
00227         }
00228         m_outerIndex[outer+1] = m_outerIndex[outer];
00229       }
00230 
00231       // here we have to handle the tricky case where the outerIndex array
00232       // starts with: [ 0 0 0 0 0 1 ...] and we are inserting in, e.g.,
00233       // the 2nd inner vector...
00234       bool isLastVec = (!(previousOuter==-1 && m_data.size()!=0))
00235                     && (size_t(m_outerIndex[outer+1]) == m_data.size());
00236 
00237       size_t startId = m_outerIndex[outer];
00238       // FIXME let's make sure sizeof(long int) == sizeof(size_t)
00239       size_t p = m_outerIndex[outer+1];
00240       ++m_outerIndex[outer+1];
00241 
00242       float reallocRatio = 1;
00243       if (m_data.allocatedSize()<=m_data.size())
00244       {
00245         // if there is no preallocated memory, let's reserve a minimum of 32 elements
00246         if (m_data.size()==0)
00247         {
00248           m_data.reserve(32);
00249         }
00250         else
00251         {
00252           // we need to reallocate the data, to reduce multiple reallocations
00253           // we use a smart resize algorithm based on the current filling ratio
00254           // in addition, we use float to avoid integers overflows
00255           float nnzEstimate = float(m_outerIndex[outer])*float(m_outerSize)/float(outer+1);
00256           reallocRatio = (nnzEstimate-float(m_data.size()))/float(m_data.size());
00257           // furthermore we bound the realloc ratio to:
00258           //   1) reduce multiple minor realloc when the matrix is almost filled
00259           //   2) avoid to allocate too much memory when the matrix is almost empty
00260           reallocRatio = (std::min)((std::max)(reallocRatio,1.5f),8.f);
00261         }
00262       }
00263       m_data.resize(m_data.size()+1,reallocRatio);
00264 
00265       if (!isLastVec)
00266       {
00267         if (previousOuter==-1)
00268         {
00269           // oops wrong guess.
00270           // let's correct the outer offsets
00271           for (Index k=0; k<=(outer+1); ++k)
00272             m_outerIndex[k] = 0;
00273           Index k=outer+1;
00274           while(m_outerIndex[k]==0)
00275             m_outerIndex[k++] = 1;
00276           while (k<=m_outerSize && m_outerIndex[k]!=0)
00277             m_outerIndex[k++]++;
00278           p = 0;
00279           --k;
00280           k = m_outerIndex[k]-1;
00281           while (k>0)
00282           {
00283             m_data.index(k) = m_data.index(k-1);
00284             m_data.value(k) = m_data.value(k-1);
00285             k--;
00286           }
00287         }
00288         else
00289         {
00290           // we are not inserting into the last inner vec
00291           // update outer indices:
00292           Index j = outer+2;
00293           while (j<=m_outerSize && m_outerIndex[j]!=0)
00294             m_outerIndex[j++]++;
00295           --j;
00296           // shift data of last vecs:
00297           Index k = m_outerIndex[j]-1;
00298           while (k>=Index(p))
00299           {
00300             m_data.index(k) = m_data.index(k-1);
00301             m_data.value(k) = m_data.value(k-1);
00302             k--;
00303           }
00304         }
00305       }
00306 
00307       while ( (p > startId) && (m_data.index(p-1) > inner) )
00308       {
00309         m_data.index(p) = m_data.index(p-1);
00310         m_data.value(p) = m_data.value(p-1);
00311         --p;
00312       }
00313 
00314       m_data.index(p) = inner;
00315       return (m_data.value(p) = 0);
00316     }
00317 
00318 
00319 
00320 
00323     inline void finalize()
00324     {
00325       Index size = static_cast<Index>(m_data.size());
00326       Index i = m_outerSize;
00327       // find the last filled column
00328       while (i>=0 && m_outerIndex[i]==0)
00329         --i;
00330       ++i;
00331       while (i<=m_outerSize)
00332       {
00333         m_outerIndex[i] = size;
00334         ++i;
00335       }
00336     }
00337 
00339     void prune(Scalar reference, RealScalar epsilon = NumTraits<RealScalar>::dummy_precision())
00340     {
00341       prune(default_prunning_func(reference,epsilon));
00342     }
00343     
00351     template<typename KeepFunc>
00352     void prune(const KeepFunc& keep = KeepFunc())
00353     {
00354       Index k = 0;
00355       for(Index j=0; j<m_outerSize; ++j)
00356       {
00357         Index previousStart = m_outerIndex[j];
00358         m_outerIndex[j] = k;
00359         Index end = m_outerIndex[j+1];
00360         for(Index i=previousStart; i<end; ++i)
00361         {
00362           if(keep(IsRowMajor?j:m_data.index(i), IsRowMajor?m_data.index(i):j, m_data.value(i)))
00363           {
00364             m_data.value(k) = m_data.value(i);
00365             m_data.index(k) = m_data.index(i);
00366             ++k;
00367           }
00368         }
00369       }
00370       m_outerIndex[m_outerSize] = k;
00371       m_data.resize(k,0);
00372     }
00373 
00377     void resize(Index rows, Index cols)
00378     {
00379       const Index outerSize = IsRowMajor ? rows : cols;
00380       m_innerSize = IsRowMajor ? cols : rows;
00381       m_data.clear();
00382       if (m_outerSize != outerSize || m_outerSize==0)
00383       {
00384         delete[] m_outerIndex;
00385         m_outerIndex = new Index [outerSize+1];
00386         m_outerSize = outerSize;
00387       }
00388       memset(m_outerIndex, 0, (m_outerSize+1)*sizeof(Index));
00389     }
00390 
00393     void resizeNonZeros(Index size)
00394     {
00395       m_data.resize(size);
00396     }
00397 
00399     inline SparseMatrix()
00400       : m_outerSize(-1), m_innerSize(0), m_outerIndex(0)
00401     {
00402       resize(0, 0);
00403     }
00404 
00406     inline SparseMatrix(Index rows, Index cols)
00407       : m_outerSize(0), m_innerSize(0), m_outerIndex(0)
00408     {
00409       resize(rows, cols);
00410     }
00411 
00413     template<typename OtherDerived>
00414     inline SparseMatrix(const SparseMatrixBase<OtherDerived>& other)
00415       : m_outerSize(0), m_innerSize(0), m_outerIndex(0)
00416     {
00417       *this = other.derived();
00418     }
00419 
00421     inline SparseMatrix(const SparseMatrix& other)
00422       : Base(), m_outerSize(0), m_innerSize(0), m_outerIndex(0)
00423     {
00424       *this = other.derived();
00425     }
00426 
00428     inline void swap(SparseMatrix& other)
00429     {
00430       //EIGEN_DBG_SPARSE(std::cout << "SparseMatrix:: swap\n");
00431       std::swap(m_outerIndex, other.m_outerIndex);
00432       std::swap(m_innerSize, other.m_innerSize);
00433       std::swap(m_outerSize, other.m_outerSize);
00434       m_data.swap(other.m_data);
00435     }
00436 
00437     inline SparseMatrix& operator=(const SparseMatrix& other)
00438     {
00439 //       std::cout << "SparseMatrix& operator=(const SparseMatrix& other)\n";
00440       if (other.isRValue())
00441       {
00442         swap(other.const_cast_derived());
00443       }
00444       else
00445       {
00446         resize(other.rows(), other.cols());
00447         memcpy(m_outerIndex, other.m_outerIndex, (m_outerSize+1)*sizeof(Index));
00448         m_data = other.m_data;
00449       }
00450       return *this;
00451     }
00452 
00453     #ifndef EIGEN_PARSED_BY_DOXYGEN
00454     template<typename Lhs, typename Rhs>
00455     inline SparseMatrix& operator=(const SparseSparseProduct<Lhs,Rhs>& product)
00456     { return Base::operator=(product); }
00457     
00458     template<typename OtherDerived>
00459     inline SparseMatrix& operator=(const ReturnByValue<OtherDerived>& other)
00460     { return Base::operator=(other); }
00461     
00462     template<typename OtherDerived>
00463     inline SparseMatrix& operator=(const EigenBase<OtherDerived>& other)
00464     { return Base::operator=(other); }
00465     #endif
00466 
00467     template<typename OtherDerived>
00468     EIGEN_DONT_INLINE SparseMatrix& operator=(const SparseMatrixBase<OtherDerived>& other)
00469     {
00470       const bool needToTranspose = (Flags & RowMajorBit) != (OtherDerived::Flags & RowMajorBit);
00471       if (needToTranspose)
00472       {
00473         // two passes algorithm:
00474         //  1 - compute the number of coeffs per dest inner vector
00475         //  2 - do the actual copy/eval
00476         // Since each coeff of the rhs has to be evaluated twice, let's evaluate it if needed
00477         typedef typename internal::nested<OtherDerived,2>::type OtherCopy;
00478         typedef typename internal::remove_all<OtherCopy>::type _OtherCopy;
00479         OtherCopy otherCopy(other.derived());
00480 
00481         resize(other.rows(), other.cols());
00482         Eigen::Map<Matrix<Index, Dynamic, 1> > (m_outerIndex,outerSize()).setZero();
00483         // pass 1
00484         // FIXME the above copy could be merged with that pass
00485         for (Index j=0; j<otherCopy.outerSize(); ++j)
00486           for (typename _OtherCopy::InnerIterator it(otherCopy, j); it; ++it)
00487             ++m_outerIndex[it.index()];
00488 
00489         // prefix sum
00490         Index count = 0;
00491         VectorXi positions(outerSize());
00492         for (Index j=0; j<outerSize(); ++j)
00493         {
00494           Index tmp = m_outerIndex[j];
00495           m_outerIndex[j] = count;
00496           positions[j] = count;
00497           count += tmp;
00498         }
00499         m_outerIndex[outerSize()] = count;
00500         // alloc
00501         m_data.resize(count);
00502         // pass 2
00503         for (Index j=0; j<otherCopy.outerSize(); ++j)
00504         {
00505           for (typename _OtherCopy::InnerIterator it(otherCopy, j); it; ++it)
00506           {
00507             Index pos = positions[it.index()]++;
00508             m_data.index(pos) = j;
00509             m_data.value(pos) = it.value();
00510           }
00511         }
00512         return *this;
00513       }
00514       else
00515       {
00516         // there is no special optimization
00517         return SparseMatrixBase<SparseMatrix>::operator=(other.derived());
00518       }
00519     }
00520 
00521     friend std::ostream & operator << (std::ostream & s, const SparseMatrix& m)
00522     {
00523       EIGEN_DBG_SPARSE(
00524         s << "Nonzero entries:\n";
00525         for (Index i=0; i<m.nonZeros(); ++i)
00526         {
00527           s << "(" << m.m_data.value(i) << "," << m.m_data.index(i) << ") ";
00528         }
00529         s << std::endl;
00530         s << std::endl;
00531         s << "Column pointers:\n";
00532         for (Index i=0; i<m.outerSize(); ++i)
00533         {
00534           s << m.m_outerIndex[i] << " ";
00535         }
00536         s << " $" << std::endl;
00537         s << std::endl;
00538       );
00539       s << static_cast<const SparseMatrixBase<SparseMatrix>&>(m);
00540       return s;
00541     }
00542 
00544     inline ~SparseMatrix()
00545     {
00546       delete[] m_outerIndex;
00547     }
00548 
00550     Scalar sum() const;
00551 
00552   public:
00553 
00559     EIGEN_DEPRECATED void startFill(Index reserveSize = 1000)
00560     {
00561       setZero();
00562       m_data.reserve(reserveSize);
00563     }
00564 
00568     EIGEN_DEPRECATED Scalar& fillrand(Index row, Index col)
00569     {
00570       return insert(row,col);
00571     }
00572 
00575     EIGEN_DEPRECATED Scalar& fill(Index row, Index col)
00576     {
00577       const Index outer = IsRowMajor ? row : col;
00578       const Index inner = IsRowMajor ? col : row;
00579 
00580       if (m_outerIndex[outer+1]==0)
00581       {
00582         // we start a new inner vector
00583         Index i = outer;
00584         while (i>=0 && m_outerIndex[i]==0)
00585         {
00586           m_outerIndex[i] = m_data.size();
00587           --i;
00588         }
00589         m_outerIndex[outer+1] = m_outerIndex[outer];
00590       }
00591       else
00592       {
00593         eigen_assert(m_data.index(m_data.size()-1)<inner && "wrong sorted insertion");
00594       }
00595 //       std::cerr << size_t(m_outerIndex[outer+1]) << " == " << m_data.size() << "\n";
00596       assert(size_t(m_outerIndex[outer+1]) == m_data.size());
00597       Index p = m_outerIndex[outer+1];
00598       ++m_outerIndex[outer+1];
00599 
00600       m_data.append(0, inner);
00601       return m_data.value(p);
00602     }
00603 
00605     EIGEN_DEPRECATED void endFill() { finalize(); }
00606     
00607 #   ifdef EIGEN_SPARSEMATRIX_PLUGIN
00608 #     include EIGEN_SPARSEMATRIX_PLUGIN
00609 #   endif
00610 
00611 private:
00612   struct default_prunning_func {
00613     default_prunning_func(Scalar ref, RealScalar eps) : reference(ref), epsilon(eps) {}
00614     inline bool operator() (const Index&, const Index&, const Scalar& value) const
00615     {
00616       return !internal::isMuchSmallerThan(value, reference, epsilon);
00617     }
00618     Scalar reference;
00619     RealScalar epsilon;
00620   };
00621 };
00622 
00623 template<typename Scalar, int _Options, typename _Index>
00624 class SparseMatrix<Scalar,_Options,_Index>::InnerIterator
00625 {
00626   public:
00627     InnerIterator(const SparseMatrix& mat, Index outer)
00628       : m_values(mat._valuePtr()), m_indices(mat._innerIndexPtr()), m_outer(outer), m_id(mat.m_outerIndex[outer]), m_end(mat.m_outerIndex[outer+1])
00629     {}
00630 
00631     inline InnerIterator& operator++() { m_id++; return *this; }
00632 
00633     inline const Scalar& value() const { return m_values[m_id]; }
00634     inline Scalar& valueRef() { return const_cast<Scalar&>(m_values[m_id]); }
00635 
00636     inline Index index() const { return m_indices[m_id]; }
00637     inline Index outer() const { return m_outer; }
00638     inline Index row() const { return IsRowMajor ? m_outer : index(); }
00639     inline Index col() const { return IsRowMajor ? index() : m_outer; }
00640 
00641     inline operator bool() const { return (m_id < m_end); }
00642 
00643   protected:
00644     const Scalar* m_values;
00645     const Index* m_indices;
00646     const Index m_outer;
00647     Index m_id;
00648     const Index m_end;
00649 };
00650 
00651 #endif // EIGEN_SPARSEMATRIX_H


libicr
Author(s): Robert Krug
autogenerated on Mon Jan 6 2014 11:33:36