如何将 C++ 指针分配给 C# 函数
How to Assign a C++ Pointer to a C# Function
我正在创建一个 C# DLL,它将用作 PLSQL Developer IDE 的插件,它是用 C++ 设计的。
我的 C# DLL 需要接受一个 C++ 指针,然后将该指针分配给一个函数或方法,以便稍后调用。
IDE 提供了构建这些插件的规范文档,但它只提供了 C++ 和 Delphi 的示例。规范文档提供了我包含在其中的更多信息
screenshot.
提供的C++例子:
void (*IDE_MenuState)(int ID, int Index, BOOL Enabled);
BOOL (*IDE_Connected)();
void (*IDE_GetConnectionInfo)(char **Username, char **Password, char **Database);
void (*IDE_GetBrowserInfo)(char **ObjectType, char **ObjectOwner, char **ObjectName);
void RegisterCallback(int Index, void *Addr)
{
switch (Index)
{
case 10 :
(void *)IDE_MenuState = Addr;
break;
case 11 :
(void *)IDE_Connected = Addr;
break;
case 12 :
(void *)IDE_GetConnectionInfo = Addr;
break;
case 13 :
(void *)IDE_GetBrowserInfo = Addr;
break;
}
}
C# 到目前为止:
我应该注意,我正在使用 Robert Gieseckes Unmanaged Exports NuGet 包来导出函数。如有必要,我可以更改它。
public bool IDE_Connected()
{
return false;
}
public void IDE_MenuState(int ID, int Index, bool Enabled)
{
}
[DllExport("add", CallingConvention = CallingConvention.Cdecl, ExportName= "RegisterCallback")]
public static void RegisterCallback(int Index, IntPtr Addr)
{
if (Index == 10)
{
// Assign IntPtr Addr to IDE_MenuState()
// Please help :)
}
if (Index == 11)
{
// Assign IntPtr Addr to IDE_Connected()
// Please help :)
}
}
如何将 C++ 指针参数分配给我的 C# 方法?
出于实际目的,无法在 运行 时更改已编译的方法。因此,无法更改 IDE_Connected()
的功能(根据您的代码示例)。
但是,您可以将您的方法声明为 delegates
(参见 here),并为每个方法创建一个静态实例。试试这个:
public unsafe class MyClass
{
delegate void IDE_MenuState(int ID, int Index, bool Enabled);
delegate bool IDE_Connected();
delegate void IDE_GetConnectionInfo(char** Username, char** Password, char** Database);
delegate void IDE_GetBrowserInfo(char** ObjectType, char** ObjectOwner, char** ObjectName);
static IDE_MenuState method_IDE_MenuState;
static IDE_Connected method_IDE_Connected;
static IDE_GetConnectionInfo method_IDE_GetConnectionInfo;
static IDE_GetBrowserInfo method_IDE_GetBrowserInfo;
public static void RegisterCallback(int Index, IntPtr Addr)
{
switch (Index)
{
case 10:
method_IDE_MenuState = (IDE_MenuState)Marshal.GetDelegateForFunctionPointer(Addr, typeof(IDE_MenuState));
break;
case 11:
method_IDE_Connected = (IDE_Connected)Marshal.GetDelegateForFunctionPointer(Addr, typeof(IDE_Connected));
break;
case 12:
method_IDE_GetConnectionInfo = (IDE_GetConnectionInfo)Marshal.GetDelegateForFunctionPointer(Addr, typeof(IDE_GetConnectionInfo));
break;
case 13:
method_IDE_GetBrowserInfo = (IDE_GetBrowserInfo)Marshal.GetDelegateForFunctionPointer(Addr, typeof(IDE_GetBrowserInfo));
break;
}
}
public static void IDE_MenuState(int ID, int Index, bool Enabled)
{
if (method_IDE_MenuState == null)
{
throw new MissingMethodException("IDE_MenuState has not been assigned pointer yet.");
}
method_IDE_MenuState(ID, Index, Enabled);
}
public static bool IDE_Connected()
{
if (method_IDE_Connected == null)
{
throw new MissingMethodException("IDE_Connected has not been assigned pointer yet.");
}
return method_IDE_Connected();
}
public static void IDE_GetConnectionInfo(char** Username, char** Password, char** Database)
{
if (method_IDE_GetConnectionInfo == null)
{
throw new MissingMethodException("IDE_GetConnectionInfo has not been assigned pointer yet.");
}
method_IDE_GetConnectionInfo(Username, Password, Database);
}
public static void IDE_GetBrowserInfo(char** ObjectType, char** ObjectOwner, char** ObjectName)
{
if (method_IDE_GetBrowserInfo == null)
{
throw new MissingMethodException("IDE_GetBrowserInfo has not been assigned pointer yet.");
}
method_IDE_GetBrowserInfo(ObjectType, ObjectOwner, ObjectName);
}
}
注意:我没有对此进行测试,但尝试更改 IDE_GetConnectionInfo
和 IDE_GetBrowserInfo
的方法签名以使用 out char[]
或更好的 out string
,而不是char**
。这将使您的 API 在 C# 中更有用。
我正在创建一个 C# DLL,它将用作 PLSQL Developer IDE 的插件,它是用 C++ 设计的。
我的 C# DLL 需要接受一个 C++ 指针,然后将该指针分配给一个函数或方法,以便稍后调用。
IDE 提供了构建这些插件的规范文档,但它只提供了 C++ 和 Delphi 的示例。规范文档提供了我包含在其中的更多信息 screenshot.
提供的C++例子:
void (*IDE_MenuState)(int ID, int Index, BOOL Enabled);
BOOL (*IDE_Connected)();
void (*IDE_GetConnectionInfo)(char **Username, char **Password, char **Database);
void (*IDE_GetBrowserInfo)(char **ObjectType, char **ObjectOwner, char **ObjectName);
void RegisterCallback(int Index, void *Addr)
{
switch (Index)
{
case 10 :
(void *)IDE_MenuState = Addr;
break;
case 11 :
(void *)IDE_Connected = Addr;
break;
case 12 :
(void *)IDE_GetConnectionInfo = Addr;
break;
case 13 :
(void *)IDE_GetBrowserInfo = Addr;
break;
}
}
C# 到目前为止:
我应该注意,我正在使用 Robert Gieseckes Unmanaged Exports NuGet 包来导出函数。如有必要,我可以更改它。
public bool IDE_Connected()
{
return false;
}
public void IDE_MenuState(int ID, int Index, bool Enabled)
{
}
[DllExport("add", CallingConvention = CallingConvention.Cdecl, ExportName= "RegisterCallback")]
public static void RegisterCallback(int Index, IntPtr Addr)
{
if (Index == 10)
{
// Assign IntPtr Addr to IDE_MenuState()
// Please help :)
}
if (Index == 11)
{
// Assign IntPtr Addr to IDE_Connected()
// Please help :)
}
}
如何将 C++ 指针参数分配给我的 C# 方法?
出于实际目的,无法在 运行 时更改已编译的方法。因此,无法更改 IDE_Connected()
的功能(根据您的代码示例)。
但是,您可以将您的方法声明为 delegates
(参见 here),并为每个方法创建一个静态实例。试试这个:
public unsafe class MyClass
{
delegate void IDE_MenuState(int ID, int Index, bool Enabled);
delegate bool IDE_Connected();
delegate void IDE_GetConnectionInfo(char** Username, char** Password, char** Database);
delegate void IDE_GetBrowserInfo(char** ObjectType, char** ObjectOwner, char** ObjectName);
static IDE_MenuState method_IDE_MenuState;
static IDE_Connected method_IDE_Connected;
static IDE_GetConnectionInfo method_IDE_GetConnectionInfo;
static IDE_GetBrowserInfo method_IDE_GetBrowserInfo;
public static void RegisterCallback(int Index, IntPtr Addr)
{
switch (Index)
{
case 10:
method_IDE_MenuState = (IDE_MenuState)Marshal.GetDelegateForFunctionPointer(Addr, typeof(IDE_MenuState));
break;
case 11:
method_IDE_Connected = (IDE_Connected)Marshal.GetDelegateForFunctionPointer(Addr, typeof(IDE_Connected));
break;
case 12:
method_IDE_GetConnectionInfo = (IDE_GetConnectionInfo)Marshal.GetDelegateForFunctionPointer(Addr, typeof(IDE_GetConnectionInfo));
break;
case 13:
method_IDE_GetBrowserInfo = (IDE_GetBrowserInfo)Marshal.GetDelegateForFunctionPointer(Addr, typeof(IDE_GetBrowserInfo));
break;
}
}
public static void IDE_MenuState(int ID, int Index, bool Enabled)
{
if (method_IDE_MenuState == null)
{
throw new MissingMethodException("IDE_MenuState has not been assigned pointer yet.");
}
method_IDE_MenuState(ID, Index, Enabled);
}
public static bool IDE_Connected()
{
if (method_IDE_Connected == null)
{
throw new MissingMethodException("IDE_Connected has not been assigned pointer yet.");
}
return method_IDE_Connected();
}
public static void IDE_GetConnectionInfo(char** Username, char** Password, char** Database)
{
if (method_IDE_GetConnectionInfo == null)
{
throw new MissingMethodException("IDE_GetConnectionInfo has not been assigned pointer yet.");
}
method_IDE_GetConnectionInfo(Username, Password, Database);
}
public static void IDE_GetBrowserInfo(char** ObjectType, char** ObjectOwner, char** ObjectName)
{
if (method_IDE_GetBrowserInfo == null)
{
throw new MissingMethodException("IDE_GetBrowserInfo has not been assigned pointer yet.");
}
method_IDE_GetBrowserInfo(ObjectType, ObjectOwner, ObjectName);
}
}
注意:我没有对此进行测试,但尝试更改 IDE_GetConnectionInfo
和 IDE_GetBrowserInfo
的方法签名以使用 out char[]
或更好的 out string
,而不是char**
。这将使您的 API 在 C# 中更有用。