QQ登录

只需一步,快速开始

登录 | 注册 | 找回密码

三维网

 找回密码
 注册

QQ登录

只需一步,快速开始

展开

通知     

查看: 8711|回复: 53
收起左侧

[已解决] CAD二次开发对二维图某些数据的统计(请我爱谁家版主再指导)

[复制链接]
发表于 2009-2-19 11:24:05 | 显示全部楼层 |阅读模式 来自: 中国北京

马上注册,结识高手,享用更多资源,轻松玩转三维网社区。

您需要 登录 才可以下载或查看,没有帐号?注册

x
新手求一代码实例教程:9 m' I+ u! h5 _! `- V
画好了一张简单的二维矩形分块图,如何按长宽尺寸分类进行统计,并导出到已有的(biao.xls)Excel表中。biao.xls表中的字段有“长”、“宽”、“块数”。VBA的代码该如何写呢?敬请高手赐教
) G9 t" `) d6 z. z% m+ S" }9 s1 O Snap1.jpg
) w# Q" K$ H  E/ o# `; G4 r( D
. W! m6 a; d' g' v[ 本帖最后由 koutx 于 2009-2-21 11:08 编辑 ]
 楼主| 发表于 2009-2-19 20:32:32 | 显示全部楼层 来自: 中国北京
如有这方面的高手能帮这个忙,也可单独联系我。koutx@sina.com
发表于 2009-2-20 10:45:24 | 显示全部楼层 来自: 中国辽宁营口
运行下面的代码之前,必须引用Microsoft Excel类库。方法是:在VBAIDE界面的“工具”菜单下点“引用”。。。# X$ n; q) I6 r) m; Z

  1. . |6 v( [" F* J3 K
  2. Sub A()+ G- K/ l2 L1 Q. C6 R
  3.     '声明一个选择集及过滤器
    * k1 a/ M% M4 Z8 }4 W
  4.     Dim SS As AcadSelectionSet, FT(0) As Integer, FD(0) As Variant& F1 C" w1 s# D* A# q! q2 m
  5.     '声明一个直线临时变量
    - w5 P3 w% M. N* g: l0 {- F9 u9 o
  6.     '声明两个直线动态数组,分别用于存放水平直线和垂直直线0 ^5 a- Y5 @& }
  7.     Dim L As AcadLine, L1() As AcadLine, L2() As AcadLine% F+ x1 R9 c1 O8 R; V( H. r
  8.     '声明一个双精度变量,用于存放计算精度。精度的用途见后面输入框中的文字
    6 b3 R7 F, G3 U& X9 }4 l
  9.     Dim 精度 As Double
    # R8 Z" e- W/ |/ ]. I) S, g
  10.     '声明循环变量
    " c5 t; \5 E. M: w$ m$ B8 R  K
  11.     Dim I As Long, J As Long, K As Long
    - v+ d8 o  W5 t* t' o
  12.     '声明四个变体变量,用于存放两相邻水平直线与两相邻垂直直线间的四个交点
    7 L& \; L3 N- n& M1 ^
  13.     '通过检查交点是否存在,鉴别该四条直线是否能围成矩形
    1 a1 V+ O  b/ P1 P) t" c
  14.     Dim P1 As Variant, P2 As Variant, P3 As Variant, P4 As Variant
    ! U, ]) V( O" W3 Z2 [4 X) A
  15.     '声明一个动态数组,用于存放查询矩形规格数量的结果
    & R& a% G9 I- v
  16.     Dim 矩形() As Double7 ^* r8 T/ [0 _6 ^! o9 T
  17.     '声明一个逻辑变量,用于条件判断
    5 X) @- V  Y- s+ p+ k$ x
  18.     Dim B As Boolean3 s" G* }: i; a6 v
  19.    
    ) b6 M& }, ^* b( I0 D! x$ r$ [( j
  20.     On Error Resume Next3 z( h) b8 ~: `' \, ^$ y+ w
  21.     With ThisDrawing
      r8 ~* h3 w# k
  22.         '输入精度
    8 p, }8 s  o# Y+ e
  23.         精度 = Val(InputBox("输入精度" & vbLf & "鉴别直线是否水平(垂直)时,如果直线与极轴的夹角(弧度)的绝对值小于该精度值,即认为该直线水平(垂直)" _3 y+ v- t9 ^0 X3 H2 l& e2 |  E# g
  24.             & vbLf & "鉴别矩形大小是否一致时,如果被比较的矩形的长(宽)与原始矩形的(长)宽之差的绝对值小于该精度值,即认为两矩形一样大小", "AutoCAD", 0.00000001))
    1 \; o/ A- W4 ]& p
  25.         '定义选择的对象为直线对象,创建选择集并由用户在屏幕上选择
    9 Q* I2 s% N1 w8 h
  26.         FT(0) = 0
    # u* Y! o0 S  [6 d. a& V0 k6 M
  27.         FD(0) = "line". B, Q* ]# c- f
  28.         Set SS = .SelectionSets.Add("SS")
    % A$ E5 W* E/ U  q4 E3 O
  29.         SS.SelectOnScreen FT, FD
    5 }! Y6 _' r9 p" q1 t3 d; l
  30.         '遍历选择集,鉴别其中的水平和垂直直线并分别存入动态数组9 N# q, X8 H! l0 }! m" p# ?9 `
  31.         For Each L In SS
    7 e1 e  }4 U& Y: q2 T* J
  32.             If L.Angle < 精度 Or L.Angle > .Utility.AngleToReal(180, acDegrees) * 2 - 精度 Or Abs(L.Angle - .Utility.AngleToReal(180, acDegrees)) < 精度 Then
    6 e2 U4 m( _1 y5 E( x1 `" R
  33.                 If UBound(L1) = -1 Then. V+ n9 h3 A) a; G$ Y" q
  34.                     ReDim L1(0), n: d0 P- r9 a4 n0 P& C
  35.                 Else" S5 Y7 d1 f+ j! @
  36.                     ReDim Preserve L1(UBound(L1) + 1)
    , w6 B8 L3 L# [% K+ W
  37.                 End If
    6 ^4 a1 R$ `- v1 l
  38.                 Set L1(UBound(L1)) = L+ ]7 e/ K; g, y; x4 \7 l) F* X
  39.             ElseIf Abs(L.Angle - .Utility.AngleToReal(90, acDegrees)) < 精度 Or Abs(L.Angle - .Utility.AngleToReal(270, acDegrees)) < 精度 Then
    5 p/ I+ y" H0 y) A9 z4 P
  40.                 If UBound(L2) = -1 Then
    & B; i; Q1 }) v; Y9 ]; J  g6 n
  41.                     ReDim L2(0)" o- i$ Y+ q! g, o  T1 B4 A$ L
  42.                 Else
    7 q4 B1 G1 x! A9 f9 U/ f/ r
  43.                     ReDim Preserve L2(UBound(L2) + 1)- e7 y; R5 ~1 L: u) S0 G8 V
  44.                 End If% \& s* a( V* p- E2 i/ x
  45.                 Set L2(UBound(L2)) = L
    ' n7 B( |+ @2 q5 t  `+ Q& u3 Z
  46.             End If5 ]* a9 j( ~8 h1 h3 P
  47.         Next
    + L1 S; x: G; E* D
  48.         '删除选择集
    , ^. @) |* S* n% {  O. Q9 d
  49.         SS.Delete. q0 e; T" x1 d# x) f0 S' y! D- W
  50.         '当水平直线和垂直直线数量均不小于2时,执行下面的代码,查询矩形规格和数量并保存
    - s- J+ W* y0 ^7 X
  51.         If UBound(L1) < 1 Or UBound(L2) < 1 Then: r2 V  N2 _2 j2 P
  52.         Else
    1 [8 R% z! k7 O3 f% |4 M. d
  53.             '水平直线数组中的直线,按起点纵坐标由小到大重新排序; g) d5 T, F% L8 T- x  p
  54.             For I = 0 To UBound(L1) - 1; j, _/ X2 d% }8 j: n! {
  55.                 For J = I + 1 To UBound(L1)
    & k# A) V" B5 w) y
  56.                     If L1(J).StartPoint(1) < L1(I).StartPoint(1) Then8 I3 r3 n0 _$ {( o' |8 d( }
  57.                         Set L = L1(I)- M0 u' t/ ^& {2 D  l8 q
  58.                         Set L1(I) = L1(J)
    3 v( g8 x. L7 c. c) \
  59.                         Set L1(J) = L7 j7 C6 R8 X' |* \0 [+ p
  60.                     End If
    2 m$ R1 K, E8 L1 O4 i* @4 t
  61.                 Next4 e4 b, l0 p. d7 G3 L
  62.             Next8 J6 Z5 R9 w" `
  63.             '垂直直线数组中的直线,按起点横坐标由小到大重新排序
    % n) z- ]1 N$ q
  64.             For I = 0 To UBound(L2) - 1
    3 j, r- d; E+ C: x# o# L/ W3 L
  65.                 For J = I + 1 To UBound(L2)1 R" M. y. R* b& z9 F" Y' D7 I
  66.                     If L2(J).StartPoint(0) < L2(I).StartPoint(0) Then$ T. O2 c+ i4 y+ h
  67.                         Set L = L2(I)
    1 e* B: H: D" S3 h2 `4 G9 X' p
  68.                         Set L2(I) = L2(J)1 ]! K' v4 I! \% g) h
  69.                         Set L2(J) = L" B& w" H4 `& s5 `; `6 a* Q" P9 o
  70.                     End If
    9 L2 K0 w0 ?: x) k) c* _; C6 b% ^1 A  O% P
  71.                 Next
    $ k$ J6 z# l. ]) ?! I4 ]
  72.             Next
    $ t7 [, y4 M; `9 s. D2 H. d
  73.             '检查相邻直线是否相交围成矩形并做进一步处理3 W! l2 u7 r% ]
  74.             For I = 0 To UBound(L1) - 1
    ! R( p4 ]0 ^" O" P  {
  75.                 For J = 0 To UBound(L2) - 1
    ( z9 [; V9 V7 n
  76.                     '获得相邻直线的交点* Z: k7 R) X; B. O7 z! i6 W  M
  77.                     P1 = L1(I).IntersectWith(L2(J), acExtendNone)
    8 H  Y2 H) }: M
  78.                     P2 = L1(I).IntersectWith(L2(J + 1), acExtendNone)
    6 I7 `9 h9 G: |/ O) f' \% q
  79.                     P3 = L1(I + 1).IntersectWith(L2(J + 1), acExtendNone)2 n  K# |& q( @% ^+ i; S' X/ U6 S  P; N
  80.                     P4 = L1(I + 1).IntersectWith(L2(J), acExtendNone)- V: M! t# N% t/ X
  81.                     '当四个交点都存在时,执行下面的代码/ p+ D3 J' A# y! Q! `
  82.                     If UBound(P1) = -1 Or UBound(P2) = -1 Or UBound(P3) = -1 Or UBound(P4) = -1 Then# J. d& R- ^& y1 R6 N. ^' L
  83.                     Else
    ! t* \% f9 B; }# K
  84.                         If UBound(矩形, 2) < 0 Then '第一个矩形直接存入数组
    5 F/ O0 B: R0 _. y9 E9 S
  85.                             ReDim 矩形(2, 0)- F$ r7 C+ ]/ D8 [% M
  86.                             矩形(0, 0) = P2(0) - P1(0)2 ?6 n2 F. y3 e+ O
  87.                             矩形(1, 0) = P3(1) - P2(1)
    / x& Q+ P1 A4 e( V2 D
  88.                             矩形(2, 0) = 1
    , v0 [" e: x2 F" K* Q  C/ @0 o
  89.                         Else '其它矩形
    & M) R8 U  u" m" S
  90.                             '检查前面存入数组的矩形中是否有相同规格
    7 d$ j! l. u! a5 d
  91.                             '如果存在,则在数组中的数量上加1,并改写逻辑变量(标记)
    2 D% r9 M. n" `1 R
  92.                             B = False% X) }: n5 |) a: h4 t% S
  93.                             For K = 0 To UBound(矩形, 2)
    - y* e0 k$ L: b+ B+ E
  94.                                 If Abs(矩形(0, K) - (P2(0) - P1(0))) < 精度 And Abs(矩形(1, K) - (P3(1) - P2(1))) < 精度 Then2 S! f3 c  \- f
  95.                                     矩形(2, K) = 矩形(2, K) + 1% `' u) O9 ~8 u) M3 e
  96.                                     B = True
    $ s8 j# ~8 }0 w9 {+ X% Y
  97.                                     Exit For
    . d. E$ m/ @3 `2 M9 t8 Y8 V
  98.                                 End If- \+ U% \  m9 Y5 M3 c! N, J
  99.                             Next$ _- b+ Y3 z& ~3 j  ^' q
  100.                             '如果数组中没有相同规格的矩形,则重定义数组,并写入新的规格、数量为1
    ( @3 s7 z% q- y
  101.                             If Not (B) Then
    " s* W% H3 c9 X: b2 N& }" X
  102.                                 ReDim Preserve 矩形(2, UBound(矩形, 2) + 1)
    $ J8 _  z) K1 J% y
  103.                                 矩形(0, UBound(矩形, 2)) = P2(0) - P1(0)
    * @2 f, u$ b7 Y. b
  104.                                 矩形(1, UBound(矩形, 2)) = P3(1) - P2(1)! Y# V$ R6 T3 I" q. P: h7 v# R
  105.                                 矩形(2, UBound(矩形, 2)) = 15 y$ i) s8 I4 @5 @. N$ J6 ^  f
  106.                             End If  j* l  u8 q$ s) j% L2 l" c% F* k
  107.                         End If
    ! a! {) p3 e- y) d
  108.                     End If& F. H- U* B$ k' r2 f, R
  109.                 Next$ ~* {/ ]4 ]9 d% F' u. s- y8 F& \
  110.             Next3 q- C7 K/ Z3 K
  111.             '如果存在矩形,把数组中的规格、数量写入Excel文档* m) x- `6 \4 K
  112.             If UBound(矩形, 2) < 0 Then
    ! d% o0 z. o: |9 ^9 ^
  113.             Else8 C" V' Y' p8 g, R( G
  114.                 '声明并启动Excel程序* a8 N- x9 s. ?, N3 \+ D
  115.                 '声明工作簿: B0 i* f/ I+ p
  116.                 Dim E As New Excel.Application, Book As Workbook
    : f6 ?1 }5 y$ ?
  117.                 '创建工作簿8 C. ~% z! ?8 D8 l. q) I
  118.                 Set Book = E.Workbooks.Add
    , b4 I/ Z& O7 M' `# y! t! V' r4 W
  119.                 '写入字段名称
    " h6 {2 _/ e; o: \) F
  120.                 Book.ActiveSheet.Cells(1, 1) = "长"
    1 v* S* \" T! f5 {' G* U4 H" S& n
  121.                 Book.ActiveSheet.Cells(1, 2) = "宽"
    ! N! t% n6 l# L+ i. u) V
  122.                 Book.ActiveSheet.Cells(1, 3) = "块数"
    , g# c) n/ m4 i% Q* c* `
  123.                 '写入矩形规格和数量
    . s$ {% F5 y' L7 f) A% }. @
  124.                 For I = 0 To UBound(矩形, 2)8 o/ [: G2 [* G+ ~( q7 F
  125.                     For J = 0 To 2
    9 e4 e* F$ u, G" J- D
  126.                         Book.ActiveSheet.Cells(I + 2, J + 1) = 矩形(J, I)
    ( I7 c( S5 P0 `) G
  127.                     Next
    5 F( I+ M4 P6 h+ f* O7 G& z
  128.                 Next' S0 @  h+ x! Y7 g
  129.                 '保存文档并退出Excel/ D- m  N% J3 V2 C
  130.                 Book.SaveAs "c:\biao.xls"
    7 d/ g% V8 u  T. q( H: }+ i
  131.                 Book.Close" Q# }& {1 d5 [% p# W' s1 w
  132.                 E.Quit$ B' U1 M) r+ N
  133.             End If, t2 d& e; u$ s/ y# G  N6 R$ y0 A
  134.         End If
    . i( M" ?* S, h6 P7 ]
  135.     End With. M" B7 f; h4 j0 N; D- B# x
  136. End Sub
    : K. c+ V& W7 C) `0 X2 o, L/ N% I7 A
复制代码
 楼主| 发表于 2009-2-20 17:41:20 | 显示全部楼层 来自: 中国北京
万分感谢我爱谁家版主。收下并试试认真研究学习。俺是新手,以后少不了讨教。
 楼主| 发表于 2009-2-21 10:58:14 | 显示全部楼层 来自: 中国北京
还得请我爱谁家版主指导:. [6 H9 \: k& _% U+ _" Z
在VBAIDE界面的“工具”菜单引用Microsoft Excel类库后,我插入一模块Model1,在之中将您的Sub A()过程代码整个贴入后,又回到CAD鼠标点选了几条水平线和几条垂直线,后通过工具菜单---宏----VBA管理器调出管理器窗口,点窗口上的宏按钮后按执行,总是不能成功导出。真不知错在哪儿。还清帮忙。1 r- L" W0 F- i9 W4 [: S
传上附件,能不能帮我看一看,改一改。多谢了。 例子.rar (19.63 KB, 下载次数: 11)
 楼主| 发表于 2009-2-23 15:50:39 | 显示全部楼层 来自: 中国北京
别沉下去呀。。
发表于 2009-2-23 21:06:05 | 显示全部楼层 来自: 中国

回复 5# koutx 的帖子

试了你的附件,可以导出EXCEL文件。不过有下列几个问题还请你自己根据实际情况考虑:4 z6 p! o! @0 F; b  Z8 B6 W4 C
1、我的代码中,保存的EXCEL文件路径是“c:\biao.xls”,在C盘根目录下,请你检查一下文件是否存在?) v% h, \7 Y  u/ s
2、我的代码只是提供一个思路,并不严谨。比如,根据水平和垂直直线判断是否围成矩形的部分,是针对你在一楼的附图编写的,只是检查了相邻直线,而未检查不相邻的直线。如果针对的是任意图形,这就是一个漏洞。实用程序应该比这段代码复杂很多。再比如,保存的文件路径,如果这个程序用于发布的话,还是使用公共对话框,由用户指定文件名和路径好一些。再比如,打开新的EXCEL进程前,可以先检查一下是否已经有EXCEL进程在运行。等等5 y' A7 n6 }- x: f# Y
3、可以改写一下代码:在声明并启动EXCEL程序的代码下面加一行
  1. E.Visible = True
复制代码
使EXCEL进程可见。删除保存文件、关闭工作簿、退出EXCEL程序的三行代码,可以在运行宏时直观地看到数据是否被写进了EXCEL表格。
 楼主| 发表于 2009-2-24 16:11:32 | 显示全部楼层 来自: 中国北京
现在的问题是If UBound(L1) = -1 Then这一句总报"下标越界"。不知是什么原因。请帮忙解释。多谢。再有我用的CAD2004,版本合适不合适?
2 E" C) y, r4 p2 i1 g- s6 J4 [
[ 本帖最后由 koutx 于 2009-2-24 16:12 编辑 ]
发表于 2009-2-25 07:42:07 | 显示全部楼层 来自: 中国

回复 8# koutx 的帖子

这个程序中使用了三个动态数组。在动态数组未被定义维数和下标之前,使用Ubound()函数检查最大下标是会出错的。因此,代码前面有一行
  1. On Error Resume Next
复制代码
这是一个错误处理语句。当一个运行时错误发生时,转到紧接着发生错误的语句之后的语句,并在此继续运行。) |  ~' [8 ?3 G
如果不使用 On Error 语句,则任何运行时错误都是致命的;也就是说,结果会导致显示错误信息并中止运行。
% {( j) R2 x8 l实际上,在程序首次运行到
  1. If UBound(L1) = -1 Then
复制代码
时,由于此时L1数组尚未被定义维数和下标,"下标越界"错误就会发生。由于程序在此之前已经执行了
  1. On Error Resume Next
复制代码
% g6 G( ]( U* x' Q
此时就会继续执行下面的
  1. ReDim L1(0)
复制代码
,然后跳过Else后面的代码段并继续向下运行。) i# \9 H1 H& j* l- y, W6 c
与此类似的代码还有
  1. If UBound(L2) = -1 Then
复制代码
  1. If UBound(L1) < 1 Or UBound(L2) < 1 Then
复制代码
  1. If UBound(矩形, 2) < 0 Then
复制代码
这也是我为什么在上面后两行代码的后面接着是Else代码段的原因。
; D3 Q% b8 p* |, i1 @$ p# b6 a, Q! B0 q7 Y
这段代码并没有使用CAD2004不支持的对象、方法和属性,在CAD2004下运行应该没有问题。
 楼主| 发表于 2009-2-25 21:49:50 | 显示全部楼层 来自: 中国北京
谢谢对On Error Resume Next的作用的详细讲解,是为了保证动态数组首次定义维数和下标语句的执行。- x& o0 G  r2 _( N& i
还有以下语句
- {! l$ H$ `! XSet SS = .SelectionSets.Add("SS")/ m0 [0 c& q5 i" {$ f( P5 p
        SS.SelectOnScreen FT, FD, L$ Y5 z( @9 N# M& @6 C4 ~8 X
在我测试时,如第一次未正常完成,在点选直线第二次执行时总出现“已存在选择集”的错误提示。不懂是什么机理?
发表于 2009-2-26 08:21:25 | 显示全部楼层 来自: 中国辽宁营口
选择集是命名对象,
  1. Set SS = .SelectionSets.Add("SS")
复制代码
括号中的"SS"就是这个新建选择集的名字。选择集是不能重名的,一个选择集在用过之后应该被删除,以免出现诸如在再次创建选择集时重名的错误,这就是
  1. SS.Delete
复制代码
的用途。当程序非正常结束,而没有删除用过的选择集时,第二次运行程序会因为文档中已存在同名的选择集在
  1. Set SS = .SelectionSets.Add("SS")
复制代码
这一行出错。( T* P7 E7 Q5 b3 S8 l
知道了原因,解决的办法就多了。比如关闭文档后重新打开,文档中就没有选择集了;再比如第二次运行程序前把代码中新建选择集的名字修改一下
  1. Set SS = .SelectionSets.Add("SSA")
复制代码
;再比如在这一行的前面加上
  1. On Error Resume Next
复制代码
 楼主| 发表于 2009-2-26 12:21:54 | 显示全部楼层 来自: 中国北京
谢谢版主精辟的讲解,胜读十天书啊!现在我先加载此工程后运行宏,再选择各直线的已能正确输出Excel。
+ o, S' q/ M" x* G- r另有一问题请教:如果在Access 的VBA代码中如何实现在调用AutoCAD的同时自动加载此工程。谢谢
0 s& {& n" t/ n1 a, g$ v2 f4 L+ f+ [0 m7 s" j. V' _/ A- `
[ 本帖最后由 koutx 于 2009-2-26 13:02 编辑 ]
发表于 2009-2-26 15:36:41 | 显示全部楼层 来自: 中国辽宁营口
  1.     Dim CAD As New AcadApplication6 W' S) d- m$ B, h* A( u
  2.     CAD.Visible = True
    1 b  d6 [- M, ?" i2 R( j- w
  3.     CAD.LoadDVB Name
复制代码
其中,Name是.dvb工程的完整路径名称字符串
 楼主| 发表于 2009-2-26 21:53:31 | 显示全部楼层 来自: 中国北京
再次谢谢,我马上试一试学一学。
 楼主| 发表于 2009-2-27 17:44:36 | 显示全部楼层 来自: 中国北京
版主提供的代码已顺利通过,现在的问题是:如果想在打开一个已有的dwg图文件时,用VBA代码自动加载如版主在3#的工程代码,如何实现?因为我在用Shell语句打开已有dwg文件时就不用再使用以下两行语句了:, I. A" \# A& H; T7 R
Dim CAD As New AcadApplication
. C; U: ]" j! s3 d- J9 T3 p& S! b    CAD.Visible = True( y, y/ S. X, @5 W$ v; j. Q
如此下面语句就不能用了
, i7 K( Q$ t% `) X. r& u    CAD.LoadDVB Name
5 C/ e* X# N5 {& n5 ~可如果只用LoadDVB Name又不管用。不知如何处理,请不吝赐教。
) [0 h8 ?1 n- V8 a1 e  r
! `4 C3 k7 S; D6 l6 {" J( i, a+ H/ l5 R2 K
注:此问题通过版主在13#中的方法即可解决了。
0 W$ a! ~9 ?$ d7 |1 f2 ^8 \
) S1 F6 ?3 J6 g4 P, {5 c[ 本帖最后由 koutx 于 2009-2-28 12:08 编辑 ]
 楼主| 发表于 2009-2-28 12:22:45 | 显示全部楼层 来自: 中国北京

请求帮助

在1#的图中,由于池子分类不是一种,在导出Excel时,如何分别将对应的类别也导出呢?见附图示。1 P/ Q/ B3 L% r6 D) q2 c
能通过填充不同的底纹来分类吗?如何分类处理?因水平太低,不能按版主的框架融会旁通,恳请版主再帮一下,百倍感谢。" z, h3 ~5 ^, k+ X0 M4 N
Snap1.jpg ! F: O& d. C# u% F' K+ B

7 `5 E- L8 x9 q& l, e[ 本帖最后由 koutx 于 2009-2-28 12:24 编辑 ]
发表于 2009-2-28 14:41:14 | 显示全部楼层 来自: 中国辽宁营口

回复 15# koutx 的帖子

LoadDVB  是 AcadApplication 对象的方法,要在VB或其它VBA中使用这个方法,就必须创建一个对CAD进程的引用,也就是代码中的 CAD 变量。, |) \( A( L$ h5 k7 p0 f: ]
打开一个已有的dwg图文件,可以使用 AcadApplication 的子对象 Documents 的 Open 方法。
" p& }4 }" w; w改写一下13楼的代码

  1. ( n& q; w) `* d) F2 O1 N4 P+ M( d
  2.     Dim CAD As New AcadApplication. F! ?/ \! l3 r) ]& U
  3.     CAD.Visible = True
    , S! G: m% X7 x) n) o; m2 l
  4.     CAD.Documents.Open .dwg文件路径* H- @& V( ^  m2 |  F7 n. ~  R
  5.     CAD.LoadDVB .dvb工程路径
    , {& t5 L0 z" o+ a
复制代码
如果一定要使用 Shell()  函数,则代码可以写成下面的形式
  1. + W$ k# k- }0 B" _- E- e
  2.     Shell acad.exe文件路径、参数和DWG文件路径
    ) j2 W6 n3 Q. x) C0 f# N; E
  3.     Dim CAD As AcadApplication' M, @% U* R$ t% [- ^* k% ?
  4.     On Error Resume Next
    ) O: C: z) g, I' j7 v$ q3 y
  5.     Do
    , b$ V3 z5 U  M& j" p+ |
  6.         Err.Clear
    3 ~; I5 q- {% d1 @' A/ w
  7.         Set CAD = GetObject(, "AutoCAD.Application")  t1 D" L- }! L1 M* F% ?+ f
  8.         DoEvents% a& ^8 ^+ m) K( @% w$ T7 e" I
  9.     Loop While Err" [, ^' [: {( Y, x. z. O
  10.     CAD.LoadDVB .dvb工程路径+ d5 b6 [, `) y0 L$ x! R
复制代码
由于 Shell()  函数并不能直接返回对打开的CAD程序的引用,还要用到 GetObject() 函数;而 Shell() 函数是以异步方式来执行其它程序的,也就是说,用 Shell() 启动的程序可能还没有完成执行过程,就已经执行到 Shell()  函数之后的语句,所以还要用循环语句反复检查确认CAD程序已经运行(如果CAD程序没有运行,GetObject() 函数会出错),才能继续执行下面 LoadDVB 这一行;如果在运行这段代码之前已经有CAD程序在运行,用 GetObject() 函数会返回运行对象表中的第一个CAD进程而不是 Shell()  函数启动的新的CAD进程,CAD变量会找错对象。。。综上所述,第二段代码与第一段代码相比有百害而无一利。

评分

参与人数 1三维币 +15 收起 理由
唐昕晨 + 15 应助

查看全部评分

 楼主| 发表于 2009-2-28 16:14:38 | 显示全部楼层 来自: 中国北京
原帖由 woaishuijia 于 2009-2-28 14:41 发表 http://www.3dportal.cn/discuz/images/common/back.gif/ C  O- L/ L+ l- f4 k9 z( {  W. {
Dim CAD As New AcadApplication: R# B# S% U, G' K* T
    CAD.Visible = True
5 o5 A, S! Q2 O# `: b    CAD.Documents.Open .dwg文件路径' s1 o- q, W( ?8 G, P: ~
    CAD.LoadDVB .dvb工程路径
0 P9 P; ^/ A0 S& I* ?9 w1 n# a1 L

  I' k$ B+ x9 t0 G非常非常感谢,不然我还只能采用版主13楼的代码调用AutoCAD后再打开dwg文件呢。只是有一点不明白:在第三句之后,用什么语句实现在加载dvb工程后,自动运行工程中的宏A呢(即不用手工在CAD中输入CADRun命令调出运行宏窗体再选择D:\CAD二次开发\Project.dvb!模块1.A后按运行)?我不知试了多少次都不能成功。希望在百忙中连同16楼中的难题给指导一下。
9 A" M# n' y6 b8 B+ s* ^1 l# r
- l: h2 {, S8 t1 K4 r' b9 ][ 本帖最后由 koutx 于 2009-2-28 16:25 编辑 ]

评分

参与人数 1三维币 +10 收起 理由
★新手★ + 10 欢迎积极讨论问题!

查看全部评分

发表于 2009-3-1 20:29:31 | 显示全部楼层 来自: 中国

回复 18# koutx 的帖子

运行宏的问题,再加上一行
  1.   {" z1 I! P+ p
  2.     CAD.RunMacro "D:\CAD二次开发\Project.dvb!模块1.A"' G4 h( d4 {4 \- D  R
复制代码
代码中的字符串是从你帖子中复制过来的,也就是“宏”对话框中的“宏名称”字符串。
/ R- X; v) f4 j* U: ?9 I0 R
" U; A3 J  @" ~6 [16楼的问题,实际上是一个如何制定规则的问题。从你的附图看到池子已不能像一楼的图那样单纯按尺寸分类,还要分析判断是A类池还是B类池,这首先需要在图上特定的位置加上适当的标记,然后再在代码中检查图形中的相应位置的标记结合矩形尺寸以区分池子的类型。我感觉,最方便的办法是在矩形内部加上单行文字,就像你的附图一样;在三楼代码的基础上做以下修改(红色部分是新增加或改动的)
' R( _- N" f7 |9 t/ o2 `& q0 U, _% j2 h
Sub A()
( m( L: Z; `$ J/ I2 U' k    '声明一个选择集及过滤器  d7 ~: k/ n4 r* W
    Dim SS As AcadSelectionSet, FT(0) As Integer, FD(0) As Variant1 X. R+ w! ~* ]$ ?: Q' [
    '声明一个直线临时变量* r6 i6 B3 f5 Y8 {
    '声明两个直线动态数组,分别用于存放水平直线和垂直直线
  D; ]6 \* f. j/ u    Dim L As AcadLine, L1() As AcadLine, L2() As AcadLine
$ S) R2 d: e. Y! M  A) H    '声明一个双精度变量,用于存放计算精度。精度的用途见后面输入框中的文字5 `" i5 W1 u3 T2 U% ]
    Dim 精度 As Double
8 O' Q& r6 K+ B( Z; R$ \    '声明循环变量( ~, R% E9 m+ T, M
    Dim I As Long, J As Long, K As Long
- l! z$ o& V( l- `1 z5 S9 `+ S, Z$ C$ A    '声明四个变体变量,用于存放两相邻水平直线与两相邻垂直直线间的四个交点  _" P6 p+ o2 J* t* E, c' b
    '通过检查交点是否存在,鉴别该四条直线是否能围成矩形
+ a6 m8 b6 I' t9 T" c$ M. N    Dim P1 As Variant, P2 As Variant, P3 As Variant, P4 As Variant
- W9 ^8 j: P+ `+ w4 n' V: O    '声明个动态数组,用于存放查询矩形规格数量的结果
5 {, R3 O, _* S1 I  P5 W    Dim 矩形() As Double'放置长、宽、数量. g- m+ {0 ~! d) T6 _% p" ^
    Dim 分类() As String'放置分类(单行文字)字符串
+ L3 Z' a1 T$ n  F5 Z* ^
    '声明一个逻辑变量,用于条件判断  F$ U: F3 t( X+ c2 i! N8 {$ X
    Dim B As Boolean
- K1 O: k# W; C2 q3 a! x   
( ]- }( M2 C& A8 ^2 ~) h. V+ O5 n    On Error Resume Next
$ S* ^$ l) ^6 {* k    With ThisDrawing
, j5 l% x3 }6 W4 c& d4 I        '输入精度
3 j; y  C% {* ]( w        精度 = Val(InputBox("输入精度" & vbLf & "鉴别直线是否水平(垂直)时,如果直线与极轴的夹角(弧度)的绝对值小于该精度值,即认为该直线水平(垂直)" _
; R, m2 `2 ^% `0 f1 ?5 ^            & vbLf & "鉴别矩形大小是否一致时,如果被比较的矩形的长(宽)与原始矩形的(长)宽之差的绝对值小于该精度值,即认为两矩形一样大小", "AutoCAD", 0.00000001))
& s' d& N; l  F' z7 x7 v/ ]; {        '定义选择的对象为直线对象,创建选择集并由用户在屏幕上选择1 R9 p5 B  T! x) ?# {8 ^
        FT(0) = 0* x8 M* x/ i, l! ?9 \: _. x2 Q1 R8 l
        FD(0) = "line"  n( j! L4 U. {/ H( Q
        Set SS = .SelectionSets.Add("SS" )1 e1 S. \/ Y8 a/ K6 d
        SS.SelectOnScreen FT, FD
) ]/ c- k4 c! O        '遍历选择集,鉴别其中的水平和垂直直线并分别存入动态数组
% f! a2 C) l8 \9 I9 l, T/ h7 v' `3 q        For Each L In SS
; I5 z2 a+ o& C9 y: D            If L.Angle < 精度 Or L.Angle > .Utility.AngleToReal(180, acDegrees) * 2 - 精度 Or Abs(L.Angle - .Utility.AngleToReal(180, acDegrees)) < 精度 Then
0 u9 X1 l2 w6 f9 L                If UBound(L1) = -1 Then( ~; U: X( `, V$ R  M4 b# ]# f
                    ReDim L1(0); X5 l9 k7 \" u! |
                Else* D1 ]% A4 Q) A+ Y
                    ReDim Preserve L1(UBound(L1) + 1)* u! h. M9 y$ e* h# s
                End If
/ N' F( ?! B  S( o8 B0 }                Set L1(UBound(L1)) = L/ p) M( a- Z) N" d
            ElseIf Abs(L.Angle - .Utility.AngleToReal(90, acDegrees)) < 精度 Or Abs(L.Angle - .Utility.AngleToReal(270, acDegrees)) < 精度 Then
2 o$ ^9 m5 v' I) @% ?  b                If UBound(L2) = -1 Then* d2 W. [: k* v3 S) F8 b
                    ReDim L2(0)
* I2 ^, `, Q6 H2 Q0 _                Else
- n' d0 v! h/ U& F- v7 @1 l                    ReDim Preserve L2(UBound(L2) + 1)
: F9 ~% K3 G  q0 ^! D                End If
0 D, R2 Q3 a, n8 n1 F2 v$ I" V                Set L2(UBound(L2)) = L8 i2 S5 n0 W8 u4 B  l) R4 X" }
            End If
: |# b9 `  l' s1 K; @( s        Next
* {3 v: k: f) D) a3 S- {        '删除选择集/ B. w( ]# q; h9 x
        SS.Delete
1 M0 h' u9 Z# E0 G: t+ W/ t9 \        '当水平直线和垂直直线数量均不小于2时,执行下面的代码,查询矩形规格和数量并保存& |' U9 u' ^2 b/ \+ s2 J
        If UBound(L1) < 1 Or UBound(L2) < 1 Then
9 y5 k% _1 ], ], n- ^) q        Else* w# @6 ]( t" p9 G" D, ^* O1 c
            '水平直线数组中的直线,按起点纵坐标由小到大重新排序/ N$ j) u2 F6 `
            For I = 0 To UBound(L1) - 1
: h0 j: v/ g( @2 p( C- ^                For J = I + 1 To UBound(L1)
& l$ S! @! Q: ]                    If L1(J).StartPoint(1) < L1(I).StartPoint(1) Then
5 `" y1 L: o# w) ^( I                        Set L = L1(I)
! A4 e* B: m* G/ |5 \) l' Y" s3 j                        Set L1(I) = L1(J)+ u' b; m4 Z* [/ c( ]1 g  C' E4 r
                        Set L1(J) = L- o9 L" J* h9 F0 [
                    End If
" y' M9 |' k) n$ G                Next& V) G) `1 Y3 C$ x
            Next
9 \6 T* R0 Q' J9 _0 ^            '垂直直线数组中的直线,按起点横坐标由小到大重新排序8 [. m1 ]0 @+ I" X9 k# V
            For I = 0 To UBound(L2) - 1, T5 D# H# I  _% x
                For J = I + 1 To UBound(L2)
! i2 O) K, l' O$ U- G/ l8 W                    If L2(J).StartPoint(0) < L2(I).StartPoint(0) Then
; ?. X. N, }8 v* I, ~; f+ A6 Z                        Set L = L2(I)1 z; o8 m8 q3 J- V8 \
                        Set L2(I) = L2(J)
* T; S6 }& S3 C- @0 }2 f                        Set L2(J) = L8 H7 M/ B9 J2 x5 V$ G+ N/ Z3 R9 y
                    End If8 @' o: @$ y$ @2 o
                Next
4 v% ?" p; S* @/ e2 J            Next
- g( |; Z" m5 ?) `. C; g; M            '为下一步选择单行文字定义选择集过滤器4 {( D! r8 }' A
            FT(0) = 05 M7 r. G) H6 b$ Y. s
            FD(0) = "text"

4 B: A3 X2 Y1 S            '检查相邻直线是否相交围成矩形并做进一步处理
9 ~& h' g) T. @: z$ m: u            For I = 0 To UBound(L1) - 1
: W) Y. \9 ~& ^5 S+ \4 a+ O+ X                For J = 0 To UBound(L2) - 1
5 Z  f- J& \- Z# r1 _  K                    '获得相邻直线的交点+ y7 q5 q7 x8 O
                    P1 = L1(I).IntersectWith(L2(J), acExtendNone)) S; F0 z& I* Y' O, Q
                    P2 = L1(I).IntersectWith(L2(J + 1), acExtendNone)  ~+ {( x% K% D  q# Q, |
                    P3 = L1(I + 1).IntersectWith(L2(J + 1), acExtendNone)5 \. O, A0 H' J. d  H" d
                    P4 = L1(I + 1).IntersectWith(L2(J), acExtendNone)
6 O: H( n/ c) y3 \; W4 l$ U9 `, ]                    '当四个交点都存在时,执行下面的代码( Q5 Y4 b+ \2 N, m$ R
                    If UBound(P1) = -1 Or UBound(P2) = -1 Or UBound(P3) = -1 Or UBound(P4) = -1 Then
) i! L& O$ {9 H/ N' F5 ~                    Else
. J- Z8 R5 M! s! R                        '新建选择集7 Q( T8 R/ _; W8 l" P2 w
                        Set SS = .SelectionSets.Add("SS" )
" p6 x1 N* n4 q0 ]% u                        '在矩形范围内框选单行文字
7 i" w! X; z0 u& r2 J                        SS.Select acSelectionSetWindow, P1, P3, FT, FD
9 o; H$ W" Q6 x/ ^8 N8 `
                        If UBound(矩形, 2) < 0 Then '第一个矩形直接存入数组
9 m' S1 C$ Q, e" J. u                            ReDim 矩形(2, 0), 分类(0)+ {  |. V# L/ y. ]5 U. i
                            矩形(0, 0) = P2(0) - P1(0)' J5 _; D8 T! p5 X$ }* x4 [
                            矩形(1, 0) = P3(1) - P2(1)
7 a( w( X  g3 K# w$ e( `& G                            矩形(2, 0) = 1% K/ r4 G; t1 L9 m
                            分类(0) = SS(0).TextString7 X& ?, g4 h) [" b, M1 R4 T
                        Else '其它矩形
" E% {8 ^; E- U( G                            '检查前面存入数组的矩形中是否有相同规格
6 J# P; ~; _/ |$ k                            '如果存在,则在数组中的数量上加1,并改写逻辑变量(标记)
) B% [' ^% N  _0 F6 C  p0 w: ~                            B = False; |8 ]- I9 {# s- e) N# U2 c  c
                            For K = 0 To UBound(矩形, 2). f( f9 O4 V% |" ?7 i* X2 K5 a5 X
                                If Abs(矩形(0, K) - (P2(0) - P1(0))) < 精度 And Abs(矩形(1, K) - (P3(1) - P2(1))) < 精度 And 分类(K) = SS(0).TextString Then, w$ _0 ^: N  U! c- K
                                    矩形(2, K) = 矩形(2, K) + 1' Z. O1 W, F9 I6 R0 s
                                    B = True
( d1 U* I6 _5 @+ u7 U9 P                                    Exit For; X4 X6 y: D) P  V# F; ?5 Z5 _
                                End If
3 s( e$ X; f% r* b9 Y4 R                            Next
9 A* Y7 H3 c  v3 m  Z( t                            '如果数组中没有相同规格的矩形,则重定义数组,并写入新的规格、数量为1
  J  P9 T& D% h. X                            If Not (B) Then1 E/ D. S" h  p8 U
                                ReDim Preserve 矩形(2, UBound(矩形, 2) + 1), 分类(UBound(分类) + 1)0 k+ E1 T3 h: b% e, J  U
                                矩形(0, UBound(矩形, 2)) = P2(0) - P1(0)2 t$ S# u* E- q0 i: z
                                矩形(1, UBound(矩形, 2)) = P3(1) - P2(1)( @$ l7 b, l7 J& R, a, g. R
                                矩形(2, UBound(矩形, 2)) = 1. [: L) N& r& l+ |
                                分类(UBound(分类)) = SS(0).TextString
: o9 J% A6 \1 [* w* _                            End If6 h% ~/ p5 L3 A( Z" i5 {- m7 r
                        End If+ `5 x& |0 z9 d* P
                        '删除选择集
# p8 E1 S2 \2 r                        SS.Delete
6 m/ `* P1 o  l. L7 c& c) P
                    End If# H1 N, r* z2 d5 h6 s
                Next
+ w$ G7 O. f6 b$ {& a: t            Next# r$ P, x3 m& n; s( l" S9 h6 p
            '如果存在矩形,把数组中的规格、数量写入Excel文档
; f2 D& T) U. b            If UBound(矩形, 2) < 0 Then( r% i6 ^& v6 M& Q( o
            Else
4 r8 T' j+ J6 D" M) t0 y                '声明并启动Excel程序
: V0 K5 }( d) D, q/ ^9 \9 }                '声明工作簿% r* t9 V9 G! y/ L
                Dim E As New Excel.Application, Book As Workbook
# H+ X0 ?7 @% ^8 F& x1 n                '创建工作簿! j/ ?# C( k. W2 x* K
                Set Book = E.Workbooks.Add
! Q# ]( d4 z6 c5 X                '写入字段名称3 S* e2 X- l. U7 l% e& C$ o
                Book.ActiveSheet.Cells(1, 1) = "分类"# B0 v. r7 B. x6 W
                Book.ActiveSheet.Cells(1, 2) = "长"9 J# K9 W- M5 x3 v& c! t% N; t! H0 K
                Book.ActiveSheet.Cells(1, 3) = "宽", F" `8 d* l% c( K' w) @, f
                Book.ActiveSheet.Cells(1, 4) = "块数"/ Z$ [4 L  O/ ~& s5 \3 Y
                '写入矩形规格和数量( G* X7 X' S/ j3 f( I# E
                For I = 0 To UBound(矩形, 2)% [" {* J8 o) A4 C. d
                    Book.ActiveSheet.Cells(I + 2, 1) = 分类(I)1 `7 `+ q" n5 p, O
                    For J = 0 To 2
. e4 K3 X0 N4 c; }# J                        Book.ActiveSheet.Cells(I + 2, J + 2) = 矩形(J, I)9 g- H' T2 b0 _
                    Next; G: B" t" K  z0 _
                Next
2 v" ]$ k1 _) D: P* j5 n7 R6 d( }                '保存文档并退出Excel5 u! o& }  \: ?. P$ \
                Book.SaveAs "c:\biao.xls"( v. q# f9 ]+ `9 O; i
                Book.Close- q7 g) c# I) {' z: K: o2 |9 Z
                E.Quit# R2 J; c1 J8 t$ X
            End If
( ^( g1 v" p9 p- I$ i1 P7 V        End If5 @+ P. C$ p: Z2 X2 p! t! n3 x
    End With/ X4 W2 M8 Z: E/ ]4 O' a& k9 D$ K
End Sub

评分

参与人数 1三维币 +10 收起 理由
★新手★ + 10 技术讨论,应助

查看全部评分

 楼主| 发表于 2009-3-1 23:35:07 | 显示全部楼层 来自: 中国北京
谢谢版主的诲人不倦,实在佩服。我又得好好学习,摸索试验几天了。遗憾的是论坛等级限制不能给版主如此精品贴子加分。 )19*(2 r9 f# L4 @; ]1 D6 n2 m* c) [( b# g
7 t4 h' y" g% f' D
[ 本帖最后由 koutx 于 2009-3-2 08:55 编辑 ]
 楼主| 发表于 2009-3-2 08:46:37 | 显示全部楼层 来自: 中国北京
Dim CAD As New AcadApplication
- P8 b) \3 G+ G" _. }CAD.Visible = True
" h# ?- x9 _, LCAD.Documents.Open "D:\CAD二次开发\例子.dwg"
! o  w+ j3 a% w* C5 S; l8 x( ]CAD.LoadDVB "D:\CAD二次开发\Project.dvb"
$ ?2 X2 w5 Y: PCAD.RunMacro "D:\CAD二次开发\Project.dvb!模块1.A": W0 [2 y2 t4 A6 t/ d! z8 p
% \5 ^5 y0 V( g
运行以上代码结果如下,不知错误原因,也无论如何解决不了,请版主费心再讲一下。谢谢# Q& C4 t3 \7 x, u' D
+ f2 U0 k2 ?. y0 R0 B1 N
结果是可打开例子.dwg,但出现错误,错误类型及部位说明如下:, y- P0 B7 |* E- H) G* K
1、有第4句但无第5句时出现图1红箭头指向所示的错误;但也将Project.dvb工程装载上了
9 }$ W/ Y  e" s8 o9 {! G2、加上第5句时,1、中的错误照旧。并出现如图2所示错误,该句黄色显示;并且指向第2句时显示CAD.Visible=<远程服务器不存在或不可用>的提示
7 A8 y6 g4 r; J; c Snap1.jpg $ @) r0 y' l: b7 u
& J2 M0 U  `& P8 G$ T
Snap2.jpg
7 U) a* O0 T, T3 X
' `! Z# H; K& V; }0 o[ 本帖最后由 koutx 于 2009-3-2 08:50 编辑 ]
发表于 2009-3-2 12:21:14 | 显示全部楼层 来自: 中国辽宁营口

回复 21# koutx 的帖子

第一个“错误”不是错误,是正常的。. w0 V  R; I( t6 m/ Y4 E3 Z
第二个确实是错误,问题可能出在被运行的“宏”有错误,使得这一行没法完成了。6 k- t0 O& A2 n# p7 ?
说来也巧,我刚才下载了你在5楼的DWG文件,用3楼的代码保存了DVB工程文件,在VB6.0中运行下列代码
  1. ( e1 }* C+ @1 i) @- i0 F2 W$ J% ]
  2. Dim CAD As New AcadApplication
    " w- ^' u. n0 M7 o! g) s/ \
  3. CAD.Visible = True
    $ b; X. C  g' H- a/ b  M
  4. CAD.Documents.Open "C:\Documents and Settings\Owner\桌面\例子.dwg"
    ' K; i" X& x, ?+ n) U
  5. CAD.LoadDVB "C:\Documents and Settings\Owner\桌面\Project1.dvb"
    ) t/ k4 _1 e7 `9 p( S
  6. CAD.RunMacro "C:\Documents and Settings\Owner\桌面\Project1.dvb!模块1.A"
    - P# u' F% X2 V0 Z
复制代码
结果出现和你所说一样的错误,原因是我在保存Project1.dvb文件时忘记引用EXCEL对象。 :lol:
 楼主| 发表于 2009-3-2 18:27:07 | 显示全部楼层 来自: 中国北京
我是在AccessVBA中作的,也引用了Excel对象,不知错出在哪儿?
发表于 2009-3-3 06:35:58 | 显示全部楼层 来自: 中国

回复 23# koutx 的帖子

能否把你的AccessVBA、AutoCADVBA、DWG文件都传上来?
 楼主| 发表于 2009-3-3 12:18:30 | 显示全部楼层 来自: 中国北京
出去刚回来上网看到了版主的贴子,现将我练习时的文件都打包传上来,请帮助指导。。(新的带分类的SUB A()尚未来得及试学。) 例子.rar (34.77 KB, 下载次数: 6)
发表回复
您需要登录后才可以回帖 登录 | 注册

本版积分规则


Licensed Copyright © 2016-2020 http://www.3dportal.cn/ All Rights Reserved 京 ICP备13008828号

小黑屋|手机版|Archiver|三维网 ( 京ICP备2023026364号-1 )

快速回复 返回顶部 返回列表