问题

我的问题是: C# 本身支持后期绑定 IDispatch 吗?


假装 我正在尝试自动化 Office,同时与客户安装的任何版本兼容。

在 .NET 世界中,如果您在安装了 Office 2000 的情况下进行开发,则从现在到世界末日,每个开发人员和每个客户都需要拥有 Office 2000。

在 .NET 之前的世界中,我们使用 通讯 与 Office 应用程序对话。

例如:

1)使用与版本无关的ProgID

"Excel.Application"

其结果是:

clsid = {00024500-0000-0000-C000-000000000046}

然后使用 COM,我们要求将这些类之一实例化为一个对象:

IUnknown unk;
CoCreateInstance(
    clsid, 
    null,
    CLSCTX_INPROC_SERVER | CLSCTX_LOCAL_SERVER,
    IUnknown, 
    out unk);

现在我们开始比赛了 - 能够从我的应用程序内部使用 Excel。当然,如果 真的 你想使用该对象,你必须调用有某种调用方法的方法。

我们 可以 掌握各种 界面 声明,翻译成我们的语言。这个技术很好,因为我们得到

  • 早期绑定
  • 代码洞察
  • 编译类型语法检查

一些示例代码可能是:

Application xl = (IExcelApplication)unk;
ExcelWorkbook workbook = xl.Workbooks.Add(template, lcid);
Worksheet worksheet = workbook.ActiveSheet;

但使用接口有一个缺点:我们必须获取各种接口声明,并将其翻译成我们的语言。我们陷入了使用基于方法的调用的困境,必须指定所有参数,例如:

ExcelWorkbook workbook = xl.Workbooks.Add(template, lcid);
xl.Worksheets.Add(before, after, count, type, lcid);

在现实世界中,这已被证明具有我们愿意放弃的缺点:

  • 早期绑定
  • 代码洞察
  • 编译时语法检查

并改为使用 智能调度 后期绑定:

Variant xl = (IDispatch)unk;
Variant newWorksheet = xl.Worksheets.Add();

由于 Excel 自动化是为 VB 脚本设计的,因此可以省略许多参数,即使没有它们也不会发生重载。

笔记: 不要将我的 Excel 示例与我想要使用 IDispatch 的原因混淆。并非每个 COM 对象都是 Excel。某些 COM 对象除了通过 IDispatch 之外没有其他支持。

有帮助吗?

解决方案

相对而言,您可以在 C# 中使用后期绑定 IDispatch 绑定。

http://support.microsoft.com/kb/302902

这是一些使用 Excel 的示例。这样你就不需要添加对 Microsoft 臃肿的 PIA 的不必要的依赖:

//Create XL
Object xl = Activator.CreateInstance(Type.GetTypeFromProgID("Excel.Application"));

//Get the workbooks collection.
//   books = xl.Workbooks;
Object books = xl.GetType().InvokeMember( "Workbooks", 
      BindingFlags.GetProperty, null, xl, null);

//Add a new workbook.
//   book = books.Add();
Objet book = books.GetType().InvokeMember( "Add", 
      BindingFlags.InvokeMethod, null, books, null );

//Get the worksheets collection.
//   sheets = book.Worksheets;
Object sheets = book.GetType().InvokeMember( "Worksheets",
      BindingFlags.GetProperty, null, book, null );

Object[] parameters;

//Get the first worksheet.
//   sheet = sheets.Item[1]
parameters = new Object[1];
parameters[0] = 1;
Object sheet = sheets.GetType().InvokeMember( "Item", 
      BindingFlags.GetProperty, null, sheets, parameters );

//Get a range object that contains cell A1.
//   range = sheet.Range["A1];
parameters = new Object[2];
parameters[0] = "A1";
parameters[1] = Missing.Value;
Object range = sheet.GetType().InvokeMember( "Range",
      BindingFlags.GetProperty, null, sheet, parameters );

//Write "Hello, World!" in cell A1.
//   range.Value = "Hello, World!";
parameters = new Object[1];
parameters[0] = "Hello, World!";
objRange_Late.GetType().InvokeMember( "Value", BindingFlags.SetProperty, 
      null, range, parameters );

//Return control of Excel to the user.
//   xl.Visible = true;
//   xl.UserControl = true;
parameters = new Object[1];
parameters[0] = true;
xl.GetType().InvokeMember( "Visible", BindingFlags.SetProperty,
      null, xl, Parameters );
xl.GetType().InvokeMember( "UserControl", BindingFlags.SetProperty,
      null, xl, Parameters );

其他提示

您必须等待 C# 4.0 发布才能获得您正在寻找的后期绑定。每当我需要互操作功能时,我都会切换回 VB.Net 模式,这样我就可以利用 C# 似乎缺乏的 COM 功能。

我使用的简单方法是在 VB.Net 中创建一个类来执行 IDispatch 工作,然后公开我想要用作包装器方法的方法,然后我可以从我的 C# 代码中随意调用它们。这不是最优雅的解决方案,但在过去的几个月里它让我摆脱了一两次困境。

C# 4的 dynamic 关键字支持 IDispatch 和后期绑定。你可以阅读 Sam Ng 的 动态系列 了解更多信息

哦,C# 4 目前仅作为 CTP 提供。您必须等待 Visual Studio vNext 或使用测试版(在 Windows Server 2008 Virtual PC 上运行)才能使用它。

如果您花时间编写一个包含您想要的方法和属性的界面并添加一些属性(我从内存中写下它,所以详细信息可能不会是正确,但我发誓它对我有用...)

    using System.Runtime.Interopservices;

    [Guid("00024500-0000-0000-C000-000000000046")]
    [InterfaceType(ComInterfaceType.InterfaceIsIDispatch)]
    interface IExcel
    {
      // sample property
      string Name{get;}
      // more properties
    }

    // and somewhere else
    void main()
    {
      Object xl = Activator.CreateInstance(Type.GetTypeFromProgID("Excel.Application"));
      IExcel excel = (IExcel)xl;
      string name = xl.name
    }

如前所述,该代码不能开箱即用,它更多的是提示在 msdn 中挖掘什么内容。

正如其他人所说 - 使用 c#4 的“dynamic”关键字很酷。这是一个简单的例子 - 它比使用“InvokeMethod”简洁得多

dynamic xl = Activator.CreateInstance(Type.GetTypeFromProgID("Excel.Application"));
dynamic books = xl.Workbooks;
dynamic book = books.Add();

Console.WriteLine(books.Count);     //Writes 1

foreach (dynamic b in books)        
{
Console.WriteLine(b.Name);      //Writes "Book1"
}

嘿,伙计,我目前有2个Codeplex项目可以解决此问题。

第一个是 LateBindingApi.Excel http://excel.codeplex.com将后期绑定调用映射到众所周知的对象模型。这是以下项目的测试项目。

第二个是代码生成器 http://latebindingapi.codeplex.com该工具从 COM 类型库创建 C# 项目。生成的项目包括带有 Latebind 访问 COM 服务器的映射器对象。亮点是该工具将不同版本的 COM 类型库转换为一个项目(例如 excel 9、10、11),并使用自定义的 SupportByLibrary 属性标记所有实体。我现在已经使用此工具分析了版本 9、10、11、12、14 中的所有 Office 应用程序,并生成了一个 C# 解决方案,其可作为测试测试版使用,并在主页上提供示例代码。

许可以下: CC-BY-SA归因
不隶属于 StackOverflow
scroll top