Можно ли переслать делегата через COM-взаимодействие?
У меня есть класс:
public class A
{
public delegate void SetTextDel(string value);
public void Test()
{
SetTextDel setText = someInterface.SetText;
icom.Set(setText);
}
}
[ComVisible(true), Guid("81C99F84-AA95-41A5-868E-62FAC8FAC263"), InterfaceType(ComInterfaceType.InterfaceIsDual)]
public interface Icom
{
void Set(Delegate del);
}
[ComVisible(true)]
[Guid("6DF6B926-8EB1-4333-827F-DD814B868097")]
[ClassInterface(ClassInterfaceType.None)]
[ComDefaultInterface(typeof(Icom))]
public class B : Icom
{
Set(Delegate del)
{
del.DynamicInvoke("some text");
}
}
И я получаю исключение targetinvocation в Set(Delegate del). Есть ли лучший способ переслать делегат в качестве параметра? Или я совершил здесь какую-то ошибку, которую не увидел. То, что я пытаюсь сделать, это передать someInterface.SetText этот метод в качестве параметра. Спасибо за помощь.
1 ответ
Следующие работы:
public class A
{
public delegate void SetTextDel(string value);
void TestSetText(string value)
{
MessageBox.Show(value);
}
public void Test(Icom icom)
{
SetTextDel del = TestSetText;
icom.Set(del);
}
}
[ComVisible(true), Guid("81C99F84-AA95-41A5-868E-62FAC8FAC263"), InterfaceType(ComInterfaceType.InterfaceIsDual)]
public interface Icom
{
void Set(Delegate del);
}
[ComVisible(true)]
[Guid("6DF6B926-8EB1-4333-827F-DD814B868097")]
[ClassInterface(ClassInterfaceType.None)]
[ComDefaultInterface(typeof(Icom))]
public class B : Icom
{
public void Set(Delegate del)
{
del.DynamicInvoke("some text");
}
}
private void buttonTest_Click(object sender, EventArgs e)
{
var a = new A();
var b = new B();
a.Test(b);
}
Далее, если вы хотите передать функцию обратного вызова из C++ в Icom.Set
, тоже будет работать следующее:
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
public delegate void SetTextDel([MarshalAs(UnmanagedType.BStr)] string value);
[ComVisible(true), Guid("81C99F84-AA95-41A5-868E-62FAC8FAC263"), InterfaceType(ComInterfaceType.InterfaceIsDual)]
public interface Icom
{
void Set([MarshalAs(UnmanagedType.FunctionPtr)] SetTextDel del);
}
[ComVisible(true)]
[Guid("6DF6B926-8EB1-4333-827F-DD814B868097")]
[ClassInterface(ClassInterfaceType.None)]
[ComDefaultInterface(typeof(Icom))]
public class B : Icom
{
public void Set(SetTextDel del)
{
del("some text");
}
}
Обязательно скомпилируйте оба ваших проекта на C# и C++ как 32-битный код. Функция обратного вызова C++ должна быть объявлена так:
static HRESULT __stdcall SetTextDelCallback(BSTR value)
{
return S_OK;
}
Наконец, правильный способ реализовать это, IMO, это просто определить интерфейс обратного вызова (например, ISetCallback
ниже) и передать объект, реализующий такой интерфейс, а не делегат. ISetCallback
может быть реализован на любом языке, C# или C++:
[ComVisible(true), Guid("2FE5D78D-D9F2-4236-9626-226356BA25E7")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
public interface ISetCallback
{
void OnSetText(string value);
}
public class A : ISetCallback
{
public void OnSetText(string value) // ISetCallback
{
MessageBox.Show(value);
}
public void Test(Icom icom)
{
icom.Set(this);
}
}
[ComVisible(true), Guid("81C99F84-AA95-41A5-868E-62FAC8FAC263"), InterfaceType(ComInterfaceType.InterfaceIsDual)]
public interface Icom
{
void Set(ISetCallback callback);
}
[ComVisible(true)]
[Guid("6DF6B926-8EB1-4333-827F-DD814B868097")]
[ClassInterface(ClassInterfaceType.None)]
[ComDefaultInterface(typeof(Icom))]
public class B : Icom
{
public void Set(ISetCallback callback)
{
callback.OnSetText("some text");
}
}
private void buttonTest_Click(object sender, EventArgs e)
{
var a = new A();
var b = new B();
a.Test(b);
}