利用c#进行autocad的二次研发(二)_c#应用

2008-02-23 05:46:56来源:互联网 阅读 ()

新老客户大回馈,云服务器低至5折

大家好,今天我继续给各位介绍利用C#进行AutoCAD的二次研发。在这一讲中,主要介绍上一讲例子中存在的问题。

在上一次的例子中我是通过引用AutoCAD 2004 Type Library来进行C#和AutoCAD之间的通信,但这种方法存在两个致命的缺点。第一个缺点是每次调试程式的时候C#都要重新启动AutoCAD,假如调试的次数很多(比如跟踪错误然后调试),那么编程的效率就很低,因为启动一次CAD还是需要较长的时间。相对于第一个缺点,第二个缺点则更要命。由于.NET本身的问题,Interop.AutoCAD.dll文档(就是通过他才实现了C#和AutoCAD之间的通信)存在着一些bug,因此虽然有时您的代码是完全正确的,但C#编译器还是抛出莫名其妙的错误。那不是完蛋了吗?我曾有一阶段就因为这两个要命的东东差一点放弃了C#而想改学ObjectArx了,呵呵,但是还是运气好,我偶尔一次在网上看了一篇外国人写的文章,他专门介绍了这两个问题的解决办法。下面就来解决这两个问题。

首先来看第二个难题,按以下步骤来进行:

1. 随便用Visual Studio .NET建立一个C#应用程式,然后按照上一篇文章中的配置加入AutoCAD 2004 Type Library,然后不加入任何代码,编译您的程式。

2. 在Visual Studio .NET命令行工具下用ildasm.exe(这个工具能够在Visual Studio .NET安装光盘中找到)把Interop.AutoCAD.dll文档(这个文档在步骤1中生成的项目的Bin\Release文档夹中)编译成中间语言Interop. AutoCAD.il。注意:在步骤1中建立的项目的编译配置为Release模式。

ildasm.exe /source Interop.AutoCAD.dll /output=Interop. AutoCAD.il

又要注意了:把ildasm.exe,Interop.AutoCAD.dll放在同一目录下。

3.在记事本中打开Interop. AutoCAD.il文档,然后查找结尾是“SinkHelper”而开头为 ".class private auto ansi sealed _DAcad“的语句,把语句中的private 改为public,然后保存Interop. AutoCAD.il文档。

4.使用ilasm.exe把Interop. AutoCAD.il文档编译为Interop.AutoCAD.dll文档,同样是在Visual Studio .NET命令行工具下进行。

ilasm.exe /resource=Interop.AutoCAD.res /dll Interop.AutoCAD.il /output=Interop. AutoCAD.dll

Interop.AutoCAD.res文档是在步骤1中生成的。

5.显然您不愿意每次编写应用程式时都通过上一篇文章中介绍的方法来加入Interop. AutoCAD.dll,那太麻烦了。您能够用下面的方法来让程式自动加入该文档:找到C:\Program Files\Microsoft.NET\ Primary Interop Assemblies 文档夹,然后把上面生成的

Interop.AutoCAD.dll文档拷贝进去。

好了,第二个问题解决了,接下来看第一个。

在VBA中,编程者能够使用GetObject函数来获得当前活动的AutoCAD对象,但在C#中却没有,为了这个函数我几乎把MSDN给翻遍了,然后去各种C#论坛问各位高手,结果都没得到解决,呵呵,可能国内使用C#的人比较少吧。还是在老外的论坛上看到了一篇就是讲这个问题的文章才把这个难题给解决了。使用下面的语句就能够获得当前活动的AutoCAD对象了:

(AcadApplication)Marshal.GetActiveObject("AutoCAD.Application.16")

(对于CAD2000和CAD2002,则把16改为15)

当然以上语句必须在AutoCAD打开的情况下才能使用,否则会发生错误,对于AutoCAD没打开的情况,能够使用上一篇文章的方法来处理。完整的连接AutoCAD和C#的源程式如下所示:

using System;

using AutoCAD;

using System.Runtime.InteropServices;

namespace AcadExample

{

public class AutoCADConnector : IDisposable

{

private AcadApplication _application;

private bool _initialized;

private bool _disposed;

public AutoCADConnector()

{

try

{

// Upon creation, attempt to retrieve running instance

_application = (AcadApplication)Marshal.GetActiveObject("AutoCAD.Application.16");

}

catch

{

try

{

// Create an instance and set flag to indicate this

_application = new AcadApplicationClass();

_initialized = true;

}

catch

{

throw;

}

}

}

// If the user doesnt call Dispose, the

// garbage collector will upon destruction

~AutoCADConnector()

{

Dispose(false);

}

public AcadApplication Application

{

get

{

// Return our internal instance of AutoCAD

return _application;

}

}

// This is the user-callable version of Dispose.

// It calls our internal version and removes the

// object from the garbage collectors queue.

public void Dispose()

{

Dispose(true);

GC.SuppressFinalize(this);

}

// This version of Dispose gets called by our

// destructor.

protected virtual void Dispose(bool disposing)

{

// If we created our AutoCAD instance, call its

// Quit method to avoid leaking memory.

标签:

版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有

上一篇: 理解c#中的string类型_c#教程

下一篇: 利用c#进行autocad的二次研发(一)_c#应用