2017年2月3日 星期五

.NET C# 建立 COM元件 (2) --- 製作 Excel VBA 後期綁定元件

前一篇文章介紹如何用C#製作Excel VBA前期綁定的COM元件 .NET C# 建立 COM元件 (1) --- 製作 Excel VBA 前期綁定元件,接著將介紹如何製作Excel VBA後期綁定的COM元件。
本篇跟前一篇文章提到的方法有部分雷同,用C#製作COM元件在網路上的文章很多,但詳細說明應用的卻很少,這裡要請各位讀者特別注意其應用的做法。

請依照下面步驟逐步實作就會清楚其中的差異。
Step 1. 使用 .NET C# 建立 COM元件 (1) --- 製作 Excel VBA 前期綁定元件 文章內的C# 範例做修改如下:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Runtime.InteropServices;

namespace TestVBA
{
    [Guid("6EE994BA-95B5-4548-98CB-B87663557880"),
    InterfaceType(ComInterfaceType.InterfaceIsIDispatch)]
    public interface IVBAFunc
    {
        [DispId(1)]
        int Multiply(int a, int b);
    }

    [ComVisible(true)]
    [Guid("A4F15491-8F1B-49C3-B626-589ECBCC60E7")]
    [ClassInterface(ClassInterfaceType.None)]
    public class VBAFunc : IVBAFunc
    {        
        public int Multiply(int a, int b)
        {
            return a * b;
        }
    }
}
第13行:[DispId(1)]為函數的標識。如果有多個函數可相應的在函數前面加[DispId(2)]、[DispId(3)]。
第9、18行:GUID可使用 Visual Studio內的工具產生。

或是採用以下寫法。
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Runtime.InteropServices;

namespace TestVBA
{
    [ClassInterface(ClassInterfaceType.None)]
    public class VBAFunc
    {
        public int Multiply(int a, int b)
        {
            return a * b;
        }
    }
}
第9行:亦可以採用[ClassInterface(ClassInterfaceType.AutoDual)],不過記住一定要添加 ClassInterface 屬性,不然註冊元件時,會因為找不到DLL進入點而無法註冊成功。
第10行: public class VBAFunc,若 改為 static class VBAFunc, 也會因為找不到DLL進入點而無法註冊成功。
第12行:public int Multiply,若改為static int Multiply,VBA執行時將無法找到該函數。


Step 2. 專案屬性設定。



Step 3. 這裡要多一個專案屬性設定內容,在Signing頁籤,勾選 Sign the assembly,新增密鑰檔案.SNK,主要用途為跟Windows系統註冊COM元件,以供後續其他應用程式使用。

輸入TestVBA後,按確定(記住不要先不要輸入密碼,不然後續要使用還需要Key密碼)。


此時會在該專案的資料夾物中多一個TestVBA.snk檔案,後續向Windows系統註冊會使用到的檔案。

Step 4. 透過Regasm.exe將編譯好的TestVBA.dll註冊到Windows系統中,32位元 Regasm.exe在C:\Windows\Microsoft.NET\Framework\v4.0.30319 ,64位元在C:\Windows\Microsoft.NET\Framework64\v4.0.30319下,開啟命令提示字元並輸入以下註冊指令。
C:\WINDOWS\Microsoft.NET\Framework\v4.0.30319>regasm C:\Code\TestVBA\TestVBA\bin\Release\TestVBA.dll /codebase
取消註冊指令。
C:\WINDOWS\Microsoft.NET\Framework\v4.0.30319>regasm C:\Code\TestVBA\TestVBA\bin\Release\TestVBA.dll /unregister

PS:如果沒有做 Step 3,直接做 Step 4,會因為無TestVBA.snk可以跟Windows系統註冊,出現以下畫面。

解決方式:
方法一:
(1) 到 C:\Program Files\Microsoft SDKs\Windows\v7.0A\bin 路徑,執行以下指令建立TestVBA.snk。
sn -k C:\Code\TestVBA\TestVBA\TestVBA.snk

(2) 將 TestVBA.snk 加入TestVBA專案中。

(3) 在Signing頁籤,勾選Sign the assembly,透過 加入 TestVBA.snk,以上動作完成後進行專案編譯。

(4) 上面設定完成後再回到 Step 4 執行COM元件註冊指令。

方法二:
(1) 建立TestVBA.snk。
(2) 打開AssemblyInfo.cs,在裡面加入 [assembly:AssemblyKeyFile("MyCom.snk")] 。
(3) 上面設定完成後再回到 Step 4 執行COM元件註冊指令。

Step 5. 輸入以下Excel VBA程式碼。
Sub TestVBA()
    Dim tool as Oblect
    Set tool = CreateObject("TestVBA.VBAFunc")
    MsgBox tool.Multiply(2, 3)
    Set tool = Nothing
End Sub
第3行:使用CreateObject建立物件,有別於 .NET C# 建立 COM元件 (1) --- 製作 Excel VBA 前期綁定元件 文章內的作法。

執行結果:

以上就是使用C#製作Excel VBA後期綁定COM元件。

補充:
如果要將後期綁定元件轉成前期綁定元件,可尤以以下指令進行轉換產生 tlb,在到VBA引用設定中添加。
regasm /tlb /codebase C:\Code\TestVBA\TestVBA\bin\Release\TestVBA.dll

相關文章:
參考資料: