如何在 net.core 3.1 中连接打开的 Excel 应用程序?

Mim*_*rok 5 c# excel

在 Net.Framework 中,我可以使用下一个函数来操作 Excel 应用程序的打开实例:

using Excel = Microsoft.Office.Interop.Excel;
public static Excel.Workbook GetOpenedExcelWorkbook(string workBookName)
{
    Excel.Application xlsApp = null;
    try { xlsApp = (Excel.Application)Marshal.GetActiveObject("Excel.Application"); }
    catch { return null; }

    foreach (Excel.Workbook wb in xlsApp.Workbooks)
    {
        if (wb.Name == workBookName)
            return wb;
    }
    return null;
}
Run Code Online (Sandbox Code Playgroud)

它很方便,因为用户只需单击 Excel 工作表中的按钮即可调用所有必要的程序。而且我可以用C#编写操作excel的代码,这比VBA更有用、更强大。VBA 中的下一个代码用于通过单击 excel 中的按钮来调用 C# 控制台应用程序。它在活动 Excel 文件附近找到适当的 exe 文件,并通过 VBA 中的 Shell 命令和必要的参数调用它。

Sub Button1_Click()
    Call ExeCaller("ConsoleApp.exe", "Button1_action_arg")
End Sub


Sub ExeCaller(ExeName As String, Optional cmd As String = "")
    Dim arg As String
    If Not Dir(ExeName) = "" Then
        arg = ExeName & " " & ActiveWorkbook.Name & IIf(cmd = "", "", " " & cmd)
    Else
        Dim pathToWB As String: pathToWB = ThisWorkbook.Path
        Dim pathToExe_Release As String: pathToExe_Release = pathToWB + "\bin\Release\" + ExeName
        Dim pathToExe_Debug As String:   pathToExe_Debug = pathToWB + "\bin\Debug\" + ExeName

        Dim dr As Double, dd As Double
        If Not Dir(pathToExe_Release) = "" Then dr = FileDateTime(pathToExe_Release)
        If Not Dir(pathToExe_Debug) = "" Then dd = FileDateTime(pathToExe_Debug)
        Dim dmax As Double: dmax = IIf(dr > dd, dr, dd)

        If (dmax = 0) Then
            If Not Dir(ExeName) = "" Then
                pathToExe = ExeName
            Else
                MsgBox "File " + ExeName + "not found!", vbCritical, Error
                Exit Sub
            End If
        ElseIf dmax = dr Then
            pathToExe = pathToExe_Release
        ElseIf dmax = dd Then
            pathToExe = pathToExe_Debug
        Else

            MsgBox "File " + ExeName + "not found!", vbCritical, Error
            Exit Sub
        End If

        arg = pathToExe & " " & ActiveWorkbook.Name & IIf(cmd = "", "", " " & cmd)
    End If

    Shell arg, vbNormalFocus
End Sub
Run Code Online (Sandbox Code Playgroud)

使其成为可能的关键功能是Marshal.GetActiveObject(string ProgID)在 C# 中,可在 Net.Framework 中使用。

现在我尝试迁移到 Net.Core 3.1。可以通过以下方式在 Net.Core 中使用 Excel 对象模型: https: //github.com/dotnet/samples/tree/master/core/extensions/ExcelDemo 这样我就可以创建 Excel.Application 的新实例并通过它创建新的或打开现有的 Excel 工作簿并使用它的所有功能进行操作。但我无法连接打开 Excel 工作簿,因为 Net.Core 中没有 Marshal.GetActiveObject(string ProgID) 函数。预览版 Net.Core 5 也没有它。并且没有类似的功能。

我尝试使用 Net.Core 中提供的 Marshal.GetObjectForIUnknown(IntPtr pUnk) 来完成此操作。据我所知,Excel.Application 是一个 COM 对象,因此它具有 IUnknown 接口。我试图通过打开Excel应用程序的过程来获取这个指针,该应用程序可以通过名称找到。我使用了这个简单的代码:

public static Excel.Application GetOpenedExcelApplication()
{
    Process[] aP = Process.GetProcesses();
    foreach (Process p in aP)
    {
        string pn = p.ProcessName.ToLower();

        if (Regex.IsMatch(pn, "excel*"))
        {
            object o = Marshal.GetObjectForIUnknown(p.Handle);
            Excel.Application app = (Excel.Application)o;
            return app;
        }
    }

    return null;
}
Run Code Online (Sandbox Code Playgroud)

Marshall.GetObjectForIUnknown() 调用线路上发生 System.ExecutionEngineException。

Net.Core 中有没有办法访问打开的 Excel 工作簿?

Mim*_*rok 5

我们可以查看.NET Framework 4.X中的Marshal类的源代码,并将GetActiveObject方法实现复制到.NET Core以及.NET 5和.NET 6的项目中。它可以工作。

public static class ExMarshal
{
    internal const String OLEAUT32 = "oleaut32.dll";
    internal const String OLE32 = "ole32.dll";

    [System.Security.SecurityCritical]  // auto-generated_required
    public static Object GetActiveObject(String progID)
    {
        Object obj = null;
        Guid clsid;

        // Call CLSIDFromProgIDEx first then fall back on CLSIDFromProgID if
        // CLSIDFromProgIDEx doesn't exist.
        try
        {
            CLSIDFromProgIDEx(progID, out clsid);
        }
        //            catch
        catch (Exception)
        {
            CLSIDFromProgID(progID, out clsid);
        }

        GetActiveObject(ref clsid, IntPtr.Zero, out obj);
        return obj;
    }

    //[DllImport(Microsoft.Win32.Win32Native.OLE32, PreserveSig = false)]
    [DllImport(OLE32, PreserveSig = false)]
    [ResourceExposure(ResourceScope.None)]
    [SuppressUnmanagedCodeSecurity]
    [System.Security.SecurityCritical]  // auto-generated
    private static extern void CLSIDFromProgIDEx([MarshalAs(UnmanagedType.LPWStr)] String progId, out Guid clsid);

    //[DllImport(Microsoft.Win32.Win32Native.OLE32, PreserveSig = false)]
    [DllImport(OLE32, PreserveSig = false)]
    [ResourceExposure(ResourceScope.None)]
    [SuppressUnmanagedCodeSecurity]
    [System.Security.SecurityCritical]  // auto-generated
    private static extern void CLSIDFromProgID([MarshalAs(UnmanagedType.LPWStr)] String progId, out Guid clsid);

    //[DllImport(Microsoft.Win32.Win32Native.OLEAUT32, PreserveSig = false)]
    [DllImport(OLEAUT32, PreserveSig = false)]
    [ResourceExposure(ResourceScope.None)]
    [SuppressUnmanagedCodeSecurity]
    [System.Security.SecurityCritical]  // auto-generated
    private static extern void GetActiveObject(ref Guid rclsid, IntPtr reserved, [MarshalAs(UnmanagedType.Interface)] out Object ppunk);
}
Run Code Online (Sandbox Code Playgroud)