|
在实际的工作中,我们经常要对一些表格数据进行处理,如果先把这些表格数据通过Excel处理,再导入到AutoCAD,+ V, D* S$ S" G1 M5 }$ G
那绝对会起到事半功倍的效果。随着Automation编程技术的出现,我们可以很方便地实现这一点了,下面是个' ]* n, d2 }- }0 O
ObjectARX的例子,其功能是动态关联Excel,然后读取其数据,并将这些数据打印在文本域中。
6 L4 X; v$ {, `; d; d //动态从Excel读取数据
! ^$ [% C, B4 m0 a5 r! S/ A' Q int DynamicReadFromExcel()
& h' Z% w4 \6 h3 Y% n. J! s* U9 c {( |& U! y$ n4 }) E! c' o
//常用变量定义
6 Q3 Y/ j5 n' K- W! ^* a _Application app;5 u0 K" G, q/ M
Workbooks books;( F; N2 y9 ^2 n& a! {
_Workbook book;$ ~" M: C6 b; ~1 ?, j4 c" W
Worksheets sheets;3 A+ l, g6 R) h
_Worksheet sheet;; Z6 ~) E% \1 U4 E
Range range;
1 O. }& L; c3 K2 i+ i1 m Range iCell;9 J# j, v0 }: }! C! b4 \$ `
LPDISPATCH lpDisp;
3 U+ Q$ M0 _, ]7 Z COleVariant
( U- u3 p# F4 o, [ covTrue((short)TRUE),1 `" i4 @& r% L# ~4 [5 c
covFalse((short)FALSE),! j( d/ a3 x/ U7 t3 X
covOptional((long)DISP_E_PARAMNOTFOUND, VT_ERROR);1 f d/ }! {" ]& m# B$ U! Y9 q; |
COleVariant vResult;& O) l6 z$ k& T) k
//采用MFC方式初始化COM库,程序结束时COM库会自动释放# H. e3 I1 v' X6 G
if(!AfxOleInit())
6 T% j; T+ T5 Y7 y: t6 T7 o* \+ u {
/ w! g C9 I7 h: t MessageBox(NULL,"初始化COM支持库失败!\n无法控制Excel!", \+ Z' V- A8 |: S; q
"TrueTable",MB_IConERROR | MB_OK);7 j3 E7 ^" w/ E- k" ]4 [/ {9 V
return RTERROR;5 S, E: E3 J: I& T; _" J
}
& ]: l: z+ H8 ?& i //关联已经运行的Excel实例
3 H7 s0 O! g. K: W: i CLSID clsid;/ `1 F0 P/ w. K) B6 t: V2 J. V4 H
CLSIDFromProgID(L"Excel.Application", &clsid);9 k' `& w3 |; ~, L
IUnknown *pUnk = NULL;
+ i6 L9 S8 W# f, C3 w6 B IDispatch *pRunDisp = NULL;- V0 ?! S4 F& e4 X8 \
for(long i=1;i<=5;i++) //做5次尝试7 h' G2 Y- Y3 d
{
6 d; \* g& r( Z' @0 s HRESULT hr = GetActiveObject(clsid, NULL, (IUnknown**)&pUnk);' X( y( E# o$ M) j
if(SUCCEEDED(hr))* l( ]3 F: p" { s& j8 b) G4 K3 G
{+ L6 }" `) e {; C( t/ i
hr = pUnk->QueryInterface(IID_IDispatch, (void **)&pRunDisp);1 {! j; Y: @ o+ G7 o- r( V
break;0 j/ \- L5 I; q) y
}/ v' ? K9 R+ W5 i) {$ S
::Sleep(10);
' o6 h4 P# i& I: | }
3 x6 `, V4 C) P# G if (!pRunDisp)
1 `2 F8 {% T6 L: w* } {
$ j" m0 P' `# b. m9 x% D+ P. P ::MessageBox(NULL, "没有发现Excel!", "TrueTable", MB_ICONHAND);
& O) i+ E4 R, p3 h return RTERROR;
: l7 w/ F" ` n }
7 a# U* b* ~ n5 X6 _3 n! ? if (pUnk) pUnk->Release();
- l) i' r; V* D& b4 E0 J //关联Excel% `; P; C, Z9 D/ p \
app.AttachDispatch (pRunDisp);/ }/ a0 c7 H( G
//得到当前活跃sheet6 P. o) a% F8 m M) o# T ~
//如果有单元格正处于编辑状态中,此操作不能返回,会一直等待0 N6 W0 i5 u! i* a6 X4 S0 ?) A" K
lpDisp=app.GetActiveSheet();
2 v) q6 N8 k" X; ]+ m y* |# R if(lpDisp==NULL); b- ]( {2 h' m* I! w) _, Z( U
{
% {) W. _. y4 Y$ p; h MessageBox(NULL, "没有发现有效的表格!", \5 M" Q+ O0 P* Z% _
"TrueTable",MB_IConERROR | MB_OK);
' s- P; s1 v. n+ w. W app.ReleaseDispatch ();
8 }9 L* Y' n* b) S, N return RTERROR;) L/ M6 G* g( [' O$ O: |3 c( f
} ?; G$ ]- `* P! k& f; u9 h3 `: d3 ]
sheet.AttachDispatch(lpDisp);6 R8 X7 o4 }9 [$ |
//已经使用的行数:
" p) \7 c2 D* [0 i& M' y long row_num;& f( U/ O: w' `5 e* Z4 W
range.AttachDispatch(sheet.GetUsedRange());
% [$ o& q! \$ M4 w0 {3 n range.AttachDispatch(range.GetRows());8 Q) k3 V; k2 e L
row_num=range.GetCount();
4 i3 T* B2 u1 U( e/ X0 R x //已经使用的列数:
4 Y; i: G+ y# f7 }9 F long col_num;) T( \0 q) y# b$ Y* o _2 D
range.AttachDispatch(sheet.GetUsedRange());
" P8 c* _; G3 d% {5 e range.AttachDispatch(range.GetColumns());
& B% J8 W* |* Q% I col_num=range.GetCount();
* k; d6 B I4 w. b- H //已经使用区域的起始行、列:
( l! G$ Z3 g. O1 l0 q$ P range.AttachDispatch(sheet.GetUsedRange());
, A( ]/ w7 ?# w% ] long StartRow=range.GetRow(); //起始行0 U/ g T* y1 j1 W
long StartCol=range.GetColumn(); //起始列
% B9 B$ P I0 O5 V9 [$ X //读取sheet名
% D( D( u: k0 \& ?: s CString SheetName=sheet.GetName();
; R" ?* x' C& ~1 N3 W$ W& | //ads_printf("\n%s",SheetName);1 P6 M3 d+ O0 S
if(col_num<2 && row_num<2) //此sheet为空$ ^/ a8 P! ]. z
{
3 F# @5 x( d8 R MessageBox(NULL,"\n当前表格没有数据!", \
6 v, t! c; b' r, V5 L7 s "TrueTable",MB_IConERROR | MB_OK);
4 r0 F( A M2 o2 [. F app.ReleaseDispatch ();- X. c8 U- m% T0 E$ a5 f
return RTERROR;
3 a0 e* J& S: d: x% S/ v }3 V' y3 d6 F0 v# Z& n8 c
else
! m3 ]- W" h1 ^+ ]4 D2 I3 J; K3 l; y {# V; P) d2 |. `& C1 P
ads_printf("\n表格%s共%d行,%d列",SheetName,row_num,col_num); U; {5 H: i# @# x5 k* |8 |
}, n1 A' l5 i& L5 A
//得到全部Cells,此时,range是cells的集合4 [: D8 V6 u4 c, a3 T
range.AttachDispatch(sheet.GetCells());# u) W: [; o C3 ]+ J5 S
//读写数据了
2 w1 D+ E5 _, t4 Q6 T6 t CString cstr;
5 }8 C/ q; m3 N6 l! L% O ads_printf("\n");/ ?, G$ W+ B; J- a3 q7 v( g9 f, _
for(long i=StartRow;i<STARTROW+ROW_NUM;I++) {
- ^9 Q3 i' a; s2 W2 {# B for(long j=StartCol;j<STARTCOL+COL_NUM;J++) {
- g5 X+ }1 z. w& L) i //读取单元格文本1 C4 f' z* w4 k8 A/ b, l- I
iCell.AttachDispatch(range.GetItem (COleVariant(i),COleVariant(j)).pdispVal );
" l( G! }% J: S vResult =iCell.GetText();
3 |7 ~8 B) l. t) e cstr=vResult.bstrVal;4 m8 D7 q' U" W. a1 Z1 `+ v
//写单元格文本& f+ N- ]" W" Y N$ I/ F$ ?
ads_printf("%s ",(LPTSTR)cstr);
o" k3 R' e2 l4 v }
- O; X. N7 r* e4 ] ads_printf("\n");% y" c3 A/ R6 ]. T
}
( S8 `( e3 ^) U //释放Dispatch) w4 L+ I- e2 j+ V
iCell.ReleaseDispatch ();' d0 O+ X: A( z% h6 ^5 Y% p: E
range.ReleaseDispatch ();
7 s: u7 l: N+ B; f6 c5 A( c1 Z sheet.ReleaseDispatch ();
: ?( m9 t" c* M: S sheets.ReleaseDispatch ();5 p4 ?; g+ [( B `
book.ReleaseDispatch ();0 f3 b) l- Z' H# q6 X9 x
books.ReleaseDispatch ();4 B; |! m, }" [8 s4 |" L3 k7 t
app.ReleaseDispatch ();6 ?& Q" H4 Z% H, W0 s9 c5 \6 `
return RTNORM;/ X* T C& H+ _' i) h( V8 F
}
5 |" t' d/ R# c1 d( \) V- ]3 _1 P 如果要输出到Excel的话,关键函数就是:
5 Q1 I: _* G# _$ W2 N iCell.SetItem(COleVariant(i),COleVariant(j),COleVariant(cstr)); |
|