Как мне вызвать COM-объект VB6 из C# с динамическим, когда у него есть параметр ref?
У меня есть следующая унаследованная функция VB6, которую я хочу вызвать из C#.
Public Function CreateMiscRepayment(ByRef objMiscRepayment As MiscRepayment) As Variant
' Code that sets objMiscRepayment here
End Function
Я использую следующий код в C#, но получаю исключение:
dynamic vb6ComObject = Activator.CreateInstance(Type.GetTypeFromProgID(progId));
dynamic miscRepayment = null;
dynamic result = vb6ComObject.CreateMiscRepayment(ref miscRepayment);
Исключение составляет:
System.ArgumentException: Could not convert argument 0 for call to CreateMiscRepayment.
at System.Dynamic.ComRuntimeHelpers.CheckThrowException(Int32 hresult, ExcepInfo& excepInfo, UInt32 argErr, String message)
at CallSite.Target(Closure , CallSite , ComObject , Object& )
at CallSite.Target(Closure , CallSite , ComObject , Object& )
at CallSite.Target(Closure , CallSite , Object , Object& )
at CallSite.Target(Closure , CallSite , Object , Object& )
Application\ApplicationClasses.cs(65,0): at ApplicationClasses.CanInstantiateMiscRepayment()
Я пытался изменить ref
в out
но получите ту же ошибку. Если я опущу ref
метод выполняется без ошибок, но конечно miscRepayment
все еще null, а не содержит объект, который должен был быть передан.
Обновить
Я пробовал некоторые другие способы, включая использование VB.NET (так как он всегда был более дружественным к COM, чем C#).
С помощью следующего кода VB.NET:
Dim vb6ComObject = Activator.CreateInstance(System.Type.GetTypeFromProgID(progId))
Dim miscRepayment = Nothing
Dim result = vb6ComObject.CreateMiscRepayment(miscRepayment)
Выдает следующее похожее, но другое исключение:
System.Runtime.InteropServices.COMException: Type mismatch. (Exception from HRESULT: 0x80020005 (DISP_E_TYPEMISMATCH))
at Microsoft.VisualBasic.CompilerServices.LateBinding.LateGet(Object o, Type objType, String name, Object[] args, String[] paramnames, Boolean[] CopyBack)
at Microsoft.VisualBasic.CompilerServices.NewLateBinding.LateGet(Object Instance, Type Type, String MemberName, Object[] Arguments, String[] ArgumentNames, Type[] TypeArguments, Boolean[] CopyBack)
UnitTest1.vb(19,0): at TestProject1.UnitTest1.TestMethod1()
Интересно, если я изменю вызов в примере кода C# или VB.NET, чтобы использовать null
/Nothing
вместо miscRepayment
затем код выполняется без исключения. Я даже установил точку останова в коде COM-объекта VB6 и могу подтвердить, что код выполняется правильно с этой стороны. Очевидно, с настройкой miscRepayment
параметр для null
/Nothing
в.NET нет способа получить созданный объект. Проблема должна быть связана с сортировкой параметров.
Я также пытался использовать Type.InvokeMember
с ParameterModifier
аргумент, который отмечает miscRepayment
будучи ref
параметр, но получите следующее исключение:
System.Reflection.TargetInvocationException: Exception has been thrown by the target of an invocation. ---> System.Runtime.InteropServices.COMException: Type mismatch. (Exception from HRESULT: 0x80020005 (DISP_E_TYPEMISMATCH))
--- End of inner exception stack trace ---
at System.RuntimeType.InvokeDispMethod(String name, BindingFlags invokeAttr, Object target, Object[] args, Boolean[] byrefModifiers, Int32 culture, String[] namedParameters)
at System.RuntimeType.InvokeMember(String name, BindingFlags bindingFlags, Binder binder, Object target, Object[] providedArgs, ParameterModifier[] modifiers, CultureInfo culture, String[] namedParams)
UnitTest1.vb(18,0): at TestProject1.UnitTest1.TestMethod1()
Наконец, я попробовал следующий код VB.NET:
Dim vb6ComObject = Activator.CreateInstance(System.Type.GetTypeFromProgID(progId))
Dim args(0) As Object
Microsoft.VisualBasic.CompilerServices.LateBinding.LateCall(vb6ComObject, type, "CreateMiscRepayment", args, Nothing, New Boolean() {True})
Выдает следующее исключение:
System.Runtime.InteropServices.COMException: Type mismatch. (Exception from HRESULT: 0x80020005 (DISP_E_TYPEMISMATCH))
at Microsoft.VisualBasic.CompilerServices.LateBinding.InternalLateCall(Object o, Type objType, String name, Object[] args, String[] paramnames, Boolean[] CopyBack, Boolean IgnoreReturn)
at Microsoft.VisualBasic.CompilerServices.LateBinding.LateCall(Object o, Type objType, String name, Object[] args, String[] paramnames, Boolean[] CopyBack)
UnitTest1.vb(17,0): at TestProject1.UnitTest1.TestMethod1()
Со всем кодом, который выдает исключение, COM-объект VB6 никогда не вызывается. Код взаимодействия COM должен задыхаться при попытке маршалинга ref
параметр.
В своих поисках в Google я натолкнулся на несколько примеров использования Type.InvokeMember
, но ref
Параметры всегда для простых типов, таких как целые числа и строки.
2 ответа
Кажется, в.NET нет способа вызвать метод для COM-объекта, который принимает ref
параметр со сложным типом.
Я подал ошибку в Microsoft. Проголосуйте, если эта проблема также затрагивает вас.
Обновление 30/4/2013
Есть комментарий к сообщению об ошибке от Microsoft, в котором говорится, что оно исправлено. Нет упоминания о том, какая версия.NET была затронута, хотя.
На самом деле не ответ, а обходной путь.
Мне кажется, что когда вы меняете способ доступа к COM-объектам из dynamic
к статическому, проблема уходит. Под статическим способом я имею в виду подготовить dll для COM-объекта, используя C:\Program Files (x86)\Microsoft SDKs\Windows\v7.0A\Bin\TlbImp.exe
, Я предполагаю, что это раннее связывание.