cmat.h
上传用户:kellyonhid
上传日期:2013-10-12
资源大小:932k
文件大小:11k
源码类别:

3D图形编程

开发平台:

Visual C++

  1. // Template Numerical Toolkit (TNT) for Linear Algebra
  2. //
  3. // BETA VERSION INCOMPLETE AND SUBJECT TO CHANGE
  4. // Please see http://math.nist.gov/tnt for updates
  5. //
  6. // R. Pozo
  7. // Mathematical and Computational Sciences Division
  8. // National Institute of Standards and Technology
  9. // C compatible matrix: row-oriented, 0-based [i][j] and 1-based (i,j) indexing
  10. //
  11. #ifndef CMAT_H
  12. #define CMAT_H
  13. #include "subscrpt.h"
  14. #include "vec.h"
  15. #include <stdlib.h>
  16. #include <assert.h>
  17. #include <iostream.h>
  18. #include <strstream.h>
  19. #ifdef TNT_USE_REGIONS
  20. #include "region2d.h"
  21. #endif
  22. template <class T>
  23. class C_matrix 
  24. {
  25.   public:
  26.     typedef Subscript   size_type;
  27.     typedef         T   value_type;
  28.     typedef         T   element_type;
  29.     typedef         T*  pointer;
  30.     typedef         T*  iterator;
  31.     typedef         T&  reference;
  32.     typedef const   T*  const_iterator;
  33.     typedef const   T&  const_reference;
  34.     Subscript lbound() const { return 1;}
  35.  
  36.   protected:
  37.     Subscript m_;
  38.     Subscript n_;
  39.     Subscript mn_;      // total size
  40.     T* v_;                  
  41.     T** row_;           
  42.     T* vm1_ ;       // these point to the same data, but are 1-based 
  43.     T** rowm1_;
  44.     // internal helper function to create the array
  45.     // of row pointers
  46.     void initialize(Subscript M, Subscript N)
  47.     {
  48.         mn_ = M*N;
  49.         m_ = M;
  50.         n_ = N;
  51.         v_ = new T[mn_]; 
  52.         row_ = new T*[M];
  53.         rowm1_ = new T*[M];
  54.         assert(v_  != NULL);
  55.         assert(row_  != NULL);
  56.         assert(rowm1_ != NULL);
  57.         T* p = v_;              
  58.         vm1_ = v_ - 1;
  59.         for (Subscript i=0; i<M; i++)
  60.         {
  61.             row_[i] = p;
  62.             rowm1_[i] = p-1;
  63.             p += N ;
  64.             
  65.         }
  66.         rowm1_ -- ;     // compensate for 1-based offset
  67.     }
  68.    
  69.     void copy(const T*  v)
  70.     {
  71.         Subscript N = m_ * n_;
  72.         Subscript i;
  73. #ifdef TNT_UNROLL
  74.         Subscript Nmod4 = N & 4;
  75.         Subscript N4 = N - Nmod4;
  76.         for (i=0; i<N4; i+=4)
  77.         {
  78.             v_[i] = v[i];
  79.             v_[i+1] = v[i+1];
  80.             v_[i+2] = v[i+2];
  81.             v_[i+3] = v[i+3];
  82.         }
  83.         for (i=N4; i< N; i++)
  84.             v_[i] = v[i];
  85. #else
  86.         for (i=0; i< N; i++)
  87.             v_[i] = v[i];
  88. #endif      
  89.     }
  90.     void set(const T& val)
  91.     {
  92.         Subscript N = m_ * n_;
  93.         Subscript i;
  94. #ifdef TNT_UNROLL
  95.         Subscript Nmod4 = N & 4;
  96.         Subscript N4 = N - Nmod4;
  97.         for (i=0; i<N4; i+=4)
  98.         {
  99.             v_[i] = val;
  100.             v_[i+1] = val;
  101.             v_[i+2] = val;
  102.             v_[i+3] = val; 
  103.         }
  104.         for (i=N4; i< N; i++)
  105.             v_[i] = val;
  106. #else
  107.         for (i=0; i< N; i++)
  108.             v_[i] = val;
  109.         
  110. #endif      
  111.     }
  112.     
  113.     
  114.     void destroy()
  115.     {     
  116.         /* do nothing, if no memory has been previously allocated */
  117.         if (v_ == NULL) return ;
  118.         /* if we are here, then matrix was previously allocated */
  119.         if (v_ != NULL) delete [] (v_);     
  120.         if (row_ != NULL) delete [] (row_);
  121.         /* return rowm1_ back to original value */
  122.         rowm1_ ++;
  123.         if (rowm1_ != NULL ) delete [] (rowm1_);
  124.     }
  125.   public:
  126.     operator T**(){ return  row_; }
  127.     operator T**() const { return row_; }
  128.     Subscript size() const { return mn_; }
  129.     // constructors
  130.     C_matrix() : m_(0), n_(0), mn_(0), v_(0), row_(0), vm1_(0), rowm1_(0) {};
  131.     C_matrix(const C_matrix<T> &A)
  132.     {
  133.         initialize(A.m_, A.n_);
  134.         copy(A.v_);
  135.     }
  136.     C_matrix(Subscript M, Subscript N, const T& value = T(0))
  137.     {
  138.         initialize(M,N);
  139.         set(value);
  140.     }
  141.     C_matrix(Subscript M, Subscript N, const T* v)
  142.     {
  143.         initialize(M,N);
  144.         copy(v);
  145.     }
  146.     C_matrix(Subscript M, Subscript N, char *s)
  147.     {
  148.         initialize(M,N);
  149.         istrstream ins(s);
  150.         Subscript i, j;
  151.         for (i=0; i<M; i++)
  152.             for (j=0; j<N; j++)
  153.                 ins >> row_[i][j];
  154.     }
  155.     // destructor
  156.     //
  157.     ~C_matrix()
  158.     {
  159.         destroy();
  160.     }
  161.     // reallocating
  162.     //
  163.     C_matrix<T>& newsize(Subscript M, Subscript N)
  164.     {
  165.         if (num_rows() == M && num_cols() == N)
  166.             return *this;
  167.         destroy();
  168.         initialize(M,N);
  169.         
  170.         return *this;
  171.     }
  172.     // assignments
  173.     //
  174.     C_matrix<T>& operator=(const C_matrix<T> &A)
  175.     {
  176.         if (v_ == A.v_)
  177.             return *this;
  178.         if (m_ == A.m_  && n_ == A.n_)      // no need to re-alloc
  179.             copy(A.v_);
  180.         else
  181.         {
  182.             destroy();
  183.             initialize(A.m_, A.n_);
  184.             copy(A.v_);
  185.         }
  186.         return *this;
  187.     }
  188.         
  189.     C_matrix<T>& operator=(const T& scalar)
  190.     { 
  191.         set(scalar); 
  192.         return *this;
  193.     }
  194.     Subscript dim(Subscript d) const 
  195.     {
  196. #ifdef TNT_BOUNDS_CHECK
  197.        assert( d >= 1);
  198.         assert( d <= 2);
  199. #endif
  200.         return (d==1) ? m_ : ((d==2) ? n_ : 0); 
  201.     }
  202.     Subscript num_rows() const { return m_; }
  203.     Subscript num_cols() const { return n_; }
  204.     inline T* operator[](Subscript i)
  205.     {
  206. #ifdef TNT_BOUNDS_CHECK
  207.         assert(0<=i);
  208.         assert(i < m_) ;
  209. #endif
  210.         return row_[i];
  211.     }
  212.     inline const T* operator[](Subscript i) const
  213.     {
  214. #ifdef TNT_BOUNDS_CHECK
  215.         assert(0<=i);
  216.         assert(i < m_) ;
  217. #endif
  218.         return row_[i];
  219.     }
  220.     inline reference operator()(Subscript i)
  221.     { 
  222. #ifdef TNT_BOUNDS_CHECK
  223.         assert(1<=i);
  224.         assert(i <= mn_) ;
  225. #endif
  226.         return vm1_[i]; 
  227.     }
  228.     inline const_reference operator()(Subscript i) const
  229.     { 
  230. #ifdef TNT_BOUNDS_CHECK
  231.         assert(1<=i);
  232.         assert(i <= mn_) ;
  233. #endif
  234.         return vm1_[i]; 
  235.     }
  236.     inline reference operator()(Subscript i, Subscript j)
  237.     { 
  238. #ifdef TNT_BOUNDS_CHECK
  239.         assert(1<=i);
  240.         assert(i <= m_) ;
  241.         assert(1<=j);
  242.         assert(j <= n_);
  243. #endif
  244.         return  rowm1_[i][j]; 
  245.     }
  246.     
  247.     inline const_reference operator() (Subscript i, Subscript j) const
  248.     {
  249. #ifdef TNT_BOUNDS_CHECK
  250.         assert(1<=i);
  251.         assert(i <= m_) ;
  252.         assert(1<=j);
  253.         assert(j <= n_);
  254. #endif
  255.         return rowm1_[i][j]; 
  256.     }
  257.     friend istream& operator>>(istream &s, C_matrix<T> &A);
  258. #ifdef TNT_USE_REGIONS
  259.     typedef Region2D<C_matrix<T> > Region;
  260.     
  261.     Region operator()(const Index1D &I, const Index1D &J)
  262.     {
  263.         return Region(*this, I,J);
  264.     }
  265.     typedef const_Region2D<C_matrix<T>,T > const_Region;
  266.     const_Region operator()(const Index1D &I, const Index1D &J) const
  267.     {
  268.         return const_Region(*this, I,J);
  269.     }
  270. #endif
  271. };
  272. /* ***************************  I/O  ********************************/
  273. template <class T>
  274. ostream& operator<<(ostream &s, const C_matrix<T> &A)
  275. {
  276.     Subscript M=A.num_rows();
  277.     Subscript N=A.num_cols();
  278.     s << M << " " << N << endl;
  279.     for (Subscript i=0; i<M; i++)
  280.     {
  281.         for (Subscript j=0; j<N; j++)
  282.         {
  283.             s << A[i][j] << " ";
  284.         }
  285.         s << endl;
  286.     }
  287.     return s;
  288. }
  289. template <class T>
  290. istream& operator>>(istream &s, C_matrix<T> &A)
  291. {
  292.     Subscript M, N;
  293.     s >> M >> N;
  294.     if ( !(M == A.m_ && N == A.n_) )
  295.     {
  296.         A.destroy();
  297.         A.initialize(M,N);
  298.     }
  299.     for (Subscript i=0; i<M; i++)
  300.         for (Subscript j=0; j<N; j++)
  301.         {
  302.             s >>  A[i][j];
  303.         }
  304.     return s;
  305. }
  306. //*******************[ basic matrix algorithms ]***************************
  307. template <class T>
  308. C_matrix<T> operator+(const C_matrix<T> &A, 
  309.     const C_matrix<T> &B)
  310. {
  311.     Subscript M = A.num_rows();
  312.     Subscript N = A.num_cols();
  313.     assert(M==B.num_rows());
  314.     assert(N==B.num_cols());
  315.     C_matrix<T> tmp(M,N);
  316.     Subscript i,j;
  317.     for (i=0; i<M; i++)
  318.         for (j=0; j<N; j++)
  319.             tmp[i][j] = A[i][j] + B[i][j];
  320.     return tmp;
  321. }
  322. template <class T>
  323. C_matrix<T> operator-(const C_matrix<T> &A, 
  324.     const C_matrix<T> &B)
  325. {
  326.     Subscript M = A.num_rows();
  327.     Subscript N = A.num_cols();
  328.     assert(M==B.num_rows());
  329.     assert(N==B.num_cols());
  330.     C_matrix<T> tmp(M,N);
  331.     Subscript i,j;
  332.     for (i=0; i<M; i++)
  333.         for (j=0; j<N; j++)
  334.             tmp[i][j] = A[i][j] - B[i][j];
  335.     return tmp;
  336. }
  337. template <class T>
  338. C_matrix<T> mult_element(const C_matrix<T> &A, 
  339.     const C_matrix<T> &B)
  340. {
  341.     Subscript M = A.num_rows();
  342.     Subscript N = A.num_cols();
  343.     assert(M==B.num_rows());
  344.     assert(N==B.num_cols());
  345.     C_matrix<T> tmp(M,N);
  346.     Subscript i,j;
  347.     for (i=0; i<M; i++)
  348.         for (j=0; j<N; j++)
  349.             tmp[i][j] = A[i][j] * B[i][j];
  350.     return tmp;
  351. }
  352. template <class T>
  353. C_matrix<T> transpose(const C_matrix<T> &A)
  354. {
  355.     Subscript M = A.num_rows();
  356.     Subscript N = A.num_cols();
  357.     C_matrix<T> S(N,M);
  358.     Subscript i, j;
  359.     for (i=0; i<M; i++)
  360.         for (j=0; j<N; j++)
  361.             S[j][i] = A[i][j];
  362.     return S;
  363. }
  364.     
  365. template <class T>
  366. inline C_matrix<T> matmult(const C_matrix<T>  &A, 
  367.     const C_matrix<T> &B)
  368. {
  369. #ifdef TNT_BOUNDS_CHECK
  370.     assert(A.num_cols() == B.num_rows());
  371. #endif
  372.     Subscript M = A.num_rows();
  373.     Subscript N = A.num_cols();
  374.     Subscript K = B.num_cols();
  375.     C_matrix<T> tmp(M,K);
  376.     T sum;
  377.     for (Subscript i=0; i<M; i++)
  378.     for (Subscript k=0; k<K; k++)
  379.     {
  380.         sum = 0;
  381.         for (Subscript j=0; j<N; j++)
  382.             sum = sum +  A[i][j] * B[j][k];
  383.         tmp[i][k] = sum; 
  384.     }
  385.     return tmp;
  386. }
  387. template <class T>
  388. inline C_matrix<T> operator*(const C_matrix<T>  &A, 
  389.     const C_matrix<T> &B)
  390. {
  391.     return matmult(A,B);
  392. }
  393. template <class T>
  394. inline C_matrix<T> matmult(const C_matrix<T>  &A, const T &b)
  395. {
  396.     Subscript M = A.num_rows();
  397.     Subscript N = A.num_cols();
  398.     C_matrix<T> tmp(M,N);
  399.     for (Subscript i=0; i<M; i++)
  400.     for (Subscript j=0; j<N; j++)
  401.     {
  402.         tmp[i][j] = A[i][j] * b;
  403.     }
  404.     return tmp;
  405. }
  406. template <class T>
  407. inline C_matrix<T> operator*(const C_matrix<T>  &A, const T &b)
  408. {
  409.     return matmult(A,b);
  410. }
  411. template <class T>
  412. inline int matmult(C_matrix<T>& C, const C_matrix<T>  &A, 
  413.     const C_matrix<T> &B)
  414. {
  415.     assert(A.num_cols() == B.num_rows());
  416.     Subscript M = A.num_rows();
  417.     Subscript N = A.num_cols();
  418.     Subscript K = B.num_cols();
  419.     C.newsize(M,K);
  420.     T sum;
  421.     const T* row_i;
  422.     const T* col_k;
  423.     for (Subscript i=0; i<M; i++)
  424.     for (Subscript k=0; k<K; k++)
  425.     {
  426.         row_i  = &(A[i][0]);
  427.         col_k  = &(B[0][k]);
  428.         sum = 0;
  429.         for (Subscript j=0; j<N; j++)
  430.         {
  431.             sum  += *row_i * *col_k;
  432.             row_i++;
  433.             col_k += K;
  434.         }
  435.         C[i][k] = sum; 
  436.     }
  437.     return 0;
  438. }
  439. template <class T>
  440. Vector<T> matmult(const C_matrix<T>  &A, const Vector<T> &x)
  441. {
  442. #ifdef TNT_BOUNDS_CHECK
  443.     assert(A.num_cols() == x.dim());
  444. #endif
  445.     Subscript M = A.num_rows();
  446.     Subscript N = A.num_cols();
  447.     Vector<T> tmp(M);
  448.     T sum;
  449.     for (Subscript i=0; i<M; i++)
  450.     {
  451.         sum = 0;
  452.         const T* rowi = A[i];
  453.         for (Subscript j=0; j<N; j++)
  454.             sum = sum +  rowi[j] * x[j];
  455.         tmp[i] = sum; 
  456.     }
  457.     return tmp;
  458. }
  459. template <class T>
  460. inline Vector<T> operator*(const C_matrix<T>  &A, const Vector<T> &x)
  461. {
  462.     return matmult(A,x);
  463. }
  464. #endif
  465. template <class T>
  466. C_matrix<T> outer_prod(const Vector<T> &a, const Vector<T> &b)
  467. {
  468. #ifdef TNT_BOUNDS_CHECK
  469.     assert(a.dim() == b.dim());
  470. #endif
  471.     Subscript N = a.dim();
  472.     C_matrix<T> tmp(N,N);
  473.     for (Subscript i=0; i<N; i++) {
  474.         tmp[i][i] = a[i] * b[i];
  475.         for (Subscript j=i+1; j<N; j++) { 
  476.     tmp[i][j] = tmp[j][i] = a[i] * b[j];
  477. }
  478.     }
  479.     return tmp;
  480. }
  481. // CMAT_H