linux 编译 suitesparse,大型稀疏矩阵运算库SuiteSparse vs2010通过编译正确执行

【实例简介】SuiteSparse是世界上最优秀的系数矩阵处理工程之一。但是SuiteSparse提供的官方代码仅包含在matlab、linux环境下编译的生成文件,不能生成在windows操作系统下VS环境下的C 库函数。本文件包括一个库函数cs.cpp和一个头文件cs.h,其中的代码是移植自SuiteSparse官方代码中的Csparse原始代码,功能包括除了复数矩阵以外的所有功能,已成功在vs2010的c 环境下执行过,在毕业设计中用于求解超大型稀疏矩阵的线性方程组(也就是大型稀疏矩阵的除法)。以下是SuiteSparse的介绍。 SuiteSparse是一组C、Fortran和MATLAB函数集,用来生成空间稀疏矩阵数据。在SuiteSparse中几何多种稀疏 矩阵的处理方法,包括矩阵的LU分解,QR分解,Cholesky分解,提供了解非线性方程组、实现最小二乘法等多种函数代码。

【实例截图】

【核心代码】

#include "cs.h"

#include

#include

/* C = alpha*A beta*B */

cs *cs_add (const cs *A, const cs *B, double alpha, double beta)

{

int p, j, nz = 0, anz, *Cp, *Ci, *Bp, m, n, bnz, *w, values ;

double *x, *Bx, *Cx ;

cs *C ;

if (!CS_CSC (A) || !CS_CSC (B)) return (NULL) ; /* check inputs */

if (A->m != B->m || A->n != B->n) return (NULL) ;

m = A->m ; anz = A->p [A->n] ;

n = B->n ; Bp = B->p ; Bx = B->x ; bnz = Bp [n] ;

w = (int *)cs_calloc (m, sizeof (int)) ; /* get workspace */

values = (A->x != NULL) && (Bx != NULL) ;

x = values ? (double *)cs_malloc (m, sizeof (double)) : NULL ; /* get workspace */

C = cs_spalloc (m, n, anz bnz, values, 0) ; /* allocate result*/

if (!C || !w || (values && !x)) return (cs_done (C, w, x, 0)) ;

Cp = C->p ; Ci = C->i ; Cx = C->x ;

for (j = 0 ; j < n ; j )

{

Cp [j] = nz ; /* column j of C starts here */

nz = cs_scatter (A, j, alpha, w, x, j 1, C, nz) ; /* alpha*A(:,j)*/

nz = cs_scatter (B, j, beta, w, x, j 1, C, nz) ; /* beta*B(:,j) */

if (values) for (p = Cp [j] ; p < nz ; p ) Cx [p] = x [Ci [p]] ;

}

Cp [n] = nz ; /* finalize the last column of C */

cs_sprealloc (C, 0) ; /* remove extra space from C */

return (cs_done (C, w, x, 1)) ; /* success; free workspace, return C */

}

/* clear w */

static int cs_wclear (int mark, int lemax, int *w, int n)

{

int k ;

if (mark < 2 || (mark lemax < 0))

{

for (k = 0 ; k < n ; k ) if (w [k] != 0) w [k] = 1 ;

mark = 2 ;

}

return (mark) ; /* at this point, w [0..n-1] < mark holds */

}

/* keep off-diagonal entries; drop diagonal entries */

static int cs_diag (int i, int j, double aij, void *other) { return (i != j) ; }

/* p = amd(A A') if symmetric is true, or amd(A'A) otherwise */

int *cs_amd (int order, const cs *A) /* order 0:natural, 1:Chol, 2:LU, 3:QR */

{

cs *C, *A2, *AT ;

int *Cp, *Ci, *last, *W, *len, *nv, *next, *P, *head, *elen, *degree, *w,

*hhead, *ATp, *ATi, d, dk, dext, lemax = 0, e, elenk, eln, i, j, k, k1,

k2, k3, jlast, ln, dense, nzmax, mindeg = 0, nvi, nvj, nvk, mark, wnvi,

ok, cnz, nel = 0, p, p1, p2, p3, p4, pj, pk, pk1, pk2, pn, q, n, m, t ;

unsigned int h ;

/* --- Construct matrix C ----------------------------------------------- */

if (!CS_CSC (A) || order <= 0 || order > 3) return (NULL) ; /* check */

AT = cs_transpose (A, 0) ; /* compute A' */

if (!AT) return (NULL) ;

m = A->m ; n = A->n ;

dense = CS_MAX (16, 10 * sqrt ((double) n)) ; /* find dense threshold */

dense = CS_MIN (n-2, dense) ;

if (order == 1 && n == m)

{

C = cs_add (A, AT, 0, 0) ; /* C = A A' */

}

else if (order == 2)

{

ATp = AT->p ; /* drop dense columns from AT */

ATi = AT->i ;

for (p2 = 0, j = 0 ; j < m ; j )

{

p = ATp [j] ; /* column j of AT starts here */

ATp [j] = p2 ; /* new column j starts here */

if (ATp [j 1] - p > dense) continue ; /* skip dense col j */

for ( ; p < ATp [j 1] ; p ) ATi [p2 ] = ATi [p] ;

}

ATp [m] = p2 ; /* finalize AT */

A2 = cs_transpose (AT, 0) ; /* A2 = AT' */

C = A2 ? cs_multiply (AT, A2) : NULL ; /* C=A'*A with no dense rows */

cs_spfree (A2) ;

}

else

{

C = cs_multiply (AT, A) ; /* C=A'*A */

}

cs_spfree (AT) ;

if (!C) return (NULL) ;

cs_fkeep (C, &cs_diag, NULL) ; /* drop diagonal entries */

Cp = C->p ;

cnz = Cp [n] ;

P = (int *)cs_malloc (n 1, sizeof (int)) ; /* allocate result */

W = (int *)cs_malloc (8*(n 1), sizeof (int)) ; /* get workspace */

t = cnz cnz/5 2*n ; /* add elbow room to C */

if (!P || !W || !cs_sprealloc (C, t)) return (cs_idone (P, C, W, 0)) ;

len = W ; nv = W (n 1) ; next = W 2*(n 1) ;

head = W 3*(n 1) ; elen = W 4*(n 1) ; degree = W 5*(n 1) ;

w = W 6*(n 1) ; hhead = W 7*(n 1) ;

last = P ; /* use P as workspace for last */

/* --- Initialize quotient graph ---------------------------------------- */

for (k = 0 ; k < n ; k ) len [k] = Cp [k 1] - Cp [k] ;

len [n] = 0 ;

nzmax = C->nzmax ;

Ci = C->i ;

for (i = 0 ; i <= n ; i )

{

head [i] = -1 ; /* degree list i is empty */

last [i] = -1 ;

next [i] = -1 ;

hhead [i] = -1 ; /* hash list i is empty */

nv [i] = 1 ; /* node i is just one node */

w [i] = 1 ; /* node i is alive */

elen [i] = 0 ; /* Ek of node i is empty */

degree [i] = len [i] ; /* degree of node i */

}

mark = cs_wclear (0, 0, w, n) ; /* clear w */

elen [n] = -2 ; /* n is a dead element */

Cp [n] = -1 ; /* n is a root of assembly tree */

w [n] = 0 ; /* n is a dead element */

/* --- Initialize degree lists ------------------------------------------ */

for (i = 0 ; i < n ; i )

{

d = degree [i] ;

if (d == 0) /* node i is empty */

{

elen [i] = -2 ; /* element i is dead */

nel ;

Cp [i] = -1 ; /* i is a root of assembly tree */

w [i] = 0 ;

}

else if (d > dense) /* node i is dense */

{

nv [i] = 0 ; /* absorb i into element n */

elen [i] = -1 ; /* node i is dead */

nel ;

Cp [i] = CS_FLIP (n) ;

nv [n] ;

}

else

{

if (head [d] != -1) last [head [d]] = i ;

next [i] = head [d] ; /* put node i in degree list d */

head [d] = i ;

}

}

while (nel < n) /* while (selecting pivots) do */

{

/* --- Select node of minimum approximate degree -------------------- */

for (k = -1 ; mindeg < n && (k = head [mindeg]) == -1 ; mindeg ) ;

if (next [k] != -1) last [next [k]] = -1 ;

head [mindeg] = next [k] ; /* remove k from degree list */

elenk = elen [k] ; /* elenk = |Ek| */

nvk = nv [k] ; /* # of nodes k represents */

nel = nvk ; /* nv[k] nodes of A eliminated */

/* --- Garbage collection ------------------------------------------- */

if (elenk > 0 && cnz mindeg >= nzmax)

{

for (j = 0 ; j < n ; j )

{

if ((p = Cp [j]) >= 0) /* j is a live node or element */

{

Cp [j] = Ci [p] ; /* save first entry of object */

Ci [p] = CS_FLIP (j) ; /* first entry is now CS_FLIP(j) */

}

}

for (q = 0, p = 0 ; p < cnz ; ) /* scan all of memory */

{

if ((j = CS_FLIP (Ci [p ])) >= 0) /* found object j */

{

Ci [q] = Cp [j] ; /* restore first entry of object */

Cp [j] = q ; /* new pointer to object j */

for (k3 = 0 ; k3 < len [j]-1 ; k3 ) Ci [q ] = Ci [p ] ;

}

}

cnz = q ; /* Ci [cnz...nzmax-1] now free */

}

/* --- Construct new element ---------------------------------------- */

dk = 0 ;

nv [k] = -nvk ; /* flag k as in Lk */

p = Cp [k] ;

pk1 = (elenk == 0) ? p : cnz ; /* do in place if elen[k] == 0 */

pk2 = pk1 ;

for (k1 = 1 ; k1 <= elenk 1 ; k1 )

{

if (k1 > elenk)

{

e = k ; /* search the nodes in k */

pj = p ; /* list of nodes starts at Ci[pj]*/

ln = len [k] - elenk ; /* length of list of nodes in k */

}

else

{

e = Ci [p ] ; /* search the nodes in e */

pj = Cp [e] ;

ln = len [e] ; /* length of list of nodes in e */

}

for (k2 = 1 ; k2 <= ln ; k2 )

{

i = Ci [pj ] ;

if ((nvi = nv [i]) <= 0) continue ; /* node i dead, or seen */

dk = nvi ; /* degree[Lk] = size of node i */

nv [i] = -nvi ; /* negate nv[i] to denote i in Lk*/

Ci [pk2 ] = i ; /* place i in Lk */

if (next [i] != -1) last [next [i]] = last [i] ;

if (last [i] != -1) /* remove i from degree list */

{

next [last [i]] = next [i] ;

}

else

{

head [degree [i]] = next [i] ;

}

}

if (e != k)

{

Cp [e] = CS_FLIP (k) ; /* absorb e into k */

w [e] = 0 ; /* e is now a dead element */

}

}

if (elenk != 0) cnz = pk2 ; /* Ci [cnz...nzmax] is free */

degree [k] = dk ; /* external degree of k - |Lk\i| */

Cp [k] = pk1 ; /* element k is in Ci[pk1..pk2-1] */

len [k] = pk2 - pk1 ;

elen [k] = -2 ; /* k is now an element */

/* --- Find set differences ----------------------------------------- */

mark = cs_wclear (mark, lemax, w, n) ; /* clear w if necessary */

for (pk = pk1 ; pk < pk2 ; pk ) /* scan 1: find |Le\Lk| */

{

i = Ci [pk] ;

if ((eln = elen [i]) <= 0) continue ;/* skip if elen[i] empty */

nvi = -nv [i] ; /* nv [i] was negated */

wnvi = mark - nvi ;

for (p = Cp [i] ; p <= Cp [i] eln - 1 ; p ) /* scan Ei */

{

e = Ci [p] ;

if (w [e] >= mark)

{

w [e] -= nvi ; /* decrement |Le\Lk| */

}

else if (w [e] != 0) /* ensure e is a live element */

{

w [e] = degree [e] wnvi ; /* 1st time e seen in scan 1 */

}

}

}

/* --- Degree update ------------------------------------------------ */

for (pk = pk1 ; pk < pk2 ; pk ) /* scan2: degree update */

{

i = Ci [pk] ; /* consider node i in Lk */

p1 = Cp [i] ;

p2 = p1 elen [i] - 1 ;

pn = p1 ;

for (h = 0, d = 0, p = p1 ; p <= p2 ; p ) /* scan Ei */

{

e = Ci [p] ;

if (w [e] != 0) /* e is an unabsorbed element */

{

dext = w [e] - mark ; /* dext = |Le\Lk| */

if (dext > 0)

{

d = dext ; /* sum up the set differences */

Ci [pn ] = e ; /* keep e in Ei */

h = e ; /* compute the hash of node i */

}

else

{

Cp [e] = CS_FLIP (k) ; /* aggressive absorb. e->k */

w [e] = 0 ; /* e is a dead element */

}

}

}

elen [i] = pn - p1 1 ; /* elen[i] = |Ei| */

p3 = pn ;

p4 = p1 len [i] ;

for (p = p2 1 ; p < p4 ; p ) /* prune edges in Ai */

{

j = Ci [p] ;

if ((nvj = nv [j]) <= 0) continue ; /* node j dead or in Lk */

d = nvj ; /* degree(i) = |j| */

Ci [pn ] = j ; /* place j in node list of i */

h = j ; /* compute hash for node i */

}

if (d == 0) /* check for mass elimination */

{

Cp [i] = CS_FLIP (k) ; /* absorb i into k */

nvi = -nv [i] ;

dk -= nvi ; /* |Lk| -= |i| */

nvk = nvi ; /* |k| = nv[i] */

nel = nvi ;

nv [i] = 0 ;

elen [i] = -1 ; /* node i is dead */

}

else

{

degree [i] = CS_MIN (degree [i], d) ; /* update degree(i) */

Ci [pn] = Ci [p3] ; /* move first node to end */

Ci [p3] = Ci [p1] ; /* move 1st el. to end of Ei */

Ci [p1] = k ; /* add k as 1st element in of Ei */

len [i] = pn - p1 1 ; /* new len of adj. list of node i */

h %= n ; /* finalize hash of i */

next [i] = hhead [h] ; /* place i in hash bucket */

hhead [h] = i ;

last [i] = h ; /* save hash of i in last[i] */

}

} /* scan2 is done */

degree [k] = dk ; /* finalize |Lk| */

lemax = CS_MAX (lemax, dk) ;

mark = cs_wclear (mark lemax, lemax, w, n) ; /* clear w */

/* --- Supernode detection ------------------------------------------ */

for (pk = pk1 ; pk < pk2 ; pk )

{

i = Ci [pk] ;

if (nv [i] >= 0) continue ; /* skip if i is dead */

h = last [i] ; /* scan hash bucket of node i */

i = hhead [h] ;

hhead [h] = -1 ; /* hash bucket will be empty */

for ( ; i != -1 && next [i] != -1 ; i = next [i], mark )

{

ln = len [i] ;

eln = elen [i] ;

for (p = Cp [i] 1 ; p <= Cp [i] ln-1 ; p ) w [Ci [p]] = mark;

jlast = i ;

for (j = next [i] ; j != -1 ; ) /* compare i with all j */

{

ok = (len [j] == ln) && (elen [j] == eln) ;

for (p = Cp [j] 1 ; ok && p <= Cp [j] ln - 1 ; p )

{

if (w [Ci [p]] != mark) ok = 0 ; /* compare i and j*/

}

if (ok) /* i and j are identical */

{

Cp [j] = CS_FLIP (i) ; /* absorb j into i */

nv [i] = nv [j] ;

nv [j] = 0 ;

elen [j] = -1 ; /* node j is dead */

j = next [j] ; /* delete j from hash bucket */

next [jlast] = j ;

}

else

{

jlast = j ; /* j and i are different */

j = next [j] ;

}

}

}

}

/* --- Finalize new element------------------------------------------ */

for (p = pk1, pk = pk1 ; pk < pk2 ; pk ) /* finalize Lk */

{

i = Ci [pk] ;

if ((nvi = -nv [i]) <= 0) continue ;/* skip if i is dead */

nv [i] = nvi ; /* restore nv[i] */

d = degree [i] dk - nvi ; /* compute external degree(i) */

d = CS_MIN (d, n - nel - nvi) ;

if (head [d] != -1) last [head [d]] = i ;

next [i] = head [d] ; /* put i back in degree list */

last [i] = -1 ;

head [d] = i ;

mindeg = CS_MIN (mindeg, d) ; /* find new minimum degree */

degree [i] = d ;

Ci [p ] = i ; /* place i in Lk */

}

nv [k] = nvk ; /* # nodes absorbed into k */

if ((len [k] = p-pk1) == 0) /* length of adj list of element k*/

{

Cp [k] = -1 ; /* k is a root of the tree */

w [k] = 0 ; /* k is now a dead element */

}

if (elenk != 0) cnz = p ; /* free unused space in Lk */

}

/* --- Postordering ----------------------------------------------------- */

for (i = 0 ; i < n ; i ) Cp [i] = CS_FLIP (Cp [i]) ;/* fix assembly tree */

for (j = 0 ; j <= n ; j ) head [j] = -1 ;

for (j = n ; j >= 0 ; j--) /* place unordered nodes in lists */

{

if (nv [j] > 0) continue ; /* skip if j is an element */

next [j] = head [Cp [j]] ; /* place j in list of its parent */

head [Cp [j]] = j ;

}

for (e = n ; e >= 0 ; e--) /* place elements in lists */

{

if (nv [e] <= 0) continue ; /* skip unless e is an element */

if (Cp [e] != -1)

{

next [e] = head [Cp [e]] ; /* place e in list of its parent */

head [Cp [e]] = e ;

}

}

for (k = 0, i = 0 ; i <= n ; i ) /* postorder the assembly tree */

{

if (Cp [i] == -1) k = cs_tdfs (i, k, head, next, P, w) ;

}

return (cs_idone (P, C, W, 1)) ;

}

csn *cs_chol (const cs *A, const css *S)

{

double d, lki, *Lx, *x, *Cx ;

int top, i, p, k, n, *Li, *Lp, *cp, *pinv, *s, *c, *parent, *Cp, *Ci ;

cs *L, *C, *E ;

csn *N ;

if (!CS_CSC (A) || !S || !S->cp || !S->parent) return (NULL) ;

n = A->n ;

N = (csn *)cs_calloc (1, sizeof (csn)) ; /* allocate result */

c = (int *)cs_malloc (2*n, sizeof (int)) ; /* get int workspace */

x = (double *)cs_malloc (n, sizeof (double)) ; /* get double workspace */

cp = S->cp ; pinv = S->pinv ; parent = S->parent ;

C = pinv ? cs_symperm (A, pinv, 1) : ((cs *) A) ;

E = pinv ? C : NULL ; /* E is alias for A, or a copy E=A(p,p) */

if (!N || !c || !x || !C) return (cs_ndone (N, E, c, x, 0)) ;

s = c n ;

Cp = C->p ; Ci = C->i ; Cx = C->x ;

N->L = L = cs_spalloc (n, n, cp [n], 1, 0) ; /* allocate result */

if (!L) return (cs_ndone (N, E, c, x, 0)) ;</

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要在Windows上编译SuiteSparse,你可以按照以下步骤进行操作: 1. 首先,你需要安装适当的编译环境。推荐使用Visual Studio作为编译器。你可以从Microsoft的官方网站下载并安装适用于你的Windows版本的Visual Studio。 2. 在安装Visual Studio之后,你需要下载并安装CMake,这是一个用于管理和生成跨平台编译的工具。你可以从CMake的官方网站下载并按照说明安装。 3. 下载SuiteSparse的源代码。你可以从SuiteSparse的官方网站或GitHub页面下载源代码的压缩文件。 4. 解压下载的压缩文件,并在解压后的文件夹中创建一个名为"build"的文件夹。这将用于在其中生成编译文件。 5. 打开CMake GUI,并将SuiteSparse的源代码路径设置为源码路径。将刚刚创建的"build"文件夹路径设置为“Build the binaries”的路径。 6. 点击“Configure”按钮,然后选择你希望使用的编译器。确保选择的编译器与Visual Studio版本一致。 7. 完成配置后,点击“Generate”按钮生成VS解决方案文件。 8. 在打开的VS解决方案中,点击“生成”按钮进行编译。这将会在"build"文件夹中生成SuiteSparse的可执行文件以及其他生成文件。 9. 编译完成后,你可以在"build"文件夹中找到生成的可执行文件。你可以将这些文件复制到你希望使用SuiteSparse的项目中,并在你的代码中引用它们。 请注意,这只是一个大致的步骤指南,具体步骤可能因你使用的软件版本和配置而有所不同。在实际操作中,可能会遇到一些编译错误,需要进一步解决。建议你参考SuiteSparse官方的文档和社区论坛,寻找更详细的指导和解决方案。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值