您的位置:首页 > 编程语言 > C#

使用托管C++粘合C#和C++代码

2010-06-23 16:00 441 查看
C#在xml读写,数据库操纵,界面构造等很多方面性能卓越;C++的效率高,是底层开发的必备武器。当然在。NET平台上C++/CLI的功能也越来越强大,可是毕竟大多数人对Managed C++不太熟悉,用起来也不是太方便(毕竟掺合的东西太多了),纯粹用C++/CLI进行开发还是比较少见的。但是我们在实际开发中完全可以通过少量的Managed C++代码来粘合和包装C#和C++,真正的工作主体部分由C#和C++各尽其能的去完成,发挥各自的专长,这样一来少量的代码就可以打开任督二脉。

本篇中,首先示例讲解如何在C#代码中调用纯C++模块中的内容。

在本例中,C#界面调用C++的加法函数,实现两个字符串的相加。当然,是使用Managed C++为桥梁。实现步骤如下所示:

首先,建立一个纯C++的static library——PureCPPSLib,其中实现一个字符串相加的类。(步骤省略)

class PureCClass
{
public:
PureCClass();
~PureCClass(void);
public:
LPCTSTR getFirstName();
void setFirstName(LPCTSTR fistName);
LPCTSTR getLastName();
void setLastName(LPCTSTR lastName);
LPCTSTR joinName();
private:
CString m_FirstName;
CString m_LastName;
CString m_fullName;
};
PureCClass::PureCClass()
{
}
PureCClass::~PureCClass(void)
{
}
LPCTSTR PureCClass::getFirstName()
{
return (LPCTSTR)m_FirstName;
}
void PureCClass::setFirstName(LPCTSTR fistName)
{
m_FirstName = fistName;
}
LPCTSTR PureCClass::getLastName()
{
return (LPCTSTR)m_LastName;
}
void PureCClass::setLastName(LPCTSTR lastName)
{
m_LastName = lastName;
}
LPCTSTR PureCClass::joinName()
{
m_fullName = m_FirstName + _T(" ") +m_LastName;
return (LPCTSTR)(m_fullName);
}


第二步,如下所示建立一个托管的DYNAMIC LINK LIBRARY工程——MgdLib.



托管工程中导入PureCPPSLib的lib文件,include类PureCClass的头文件,构造一个托管类——mgClass,实现对类PureCClass的托管包装。该托管类放在namespace MgdLib 中。

namespace MgdLib {
public ref class mgClass
{
public:
mgClass();
protected:
!mgClass();
public:
~mgClass();
property String^ FirstName
{
String ^ get ();
void set (String ^str);
}
property String^ LastName
{
String ^ get ();
void set (String ^str);
}
String^ JoinName();
private:
PureCClass* m_pImpObj;
};
}
namespace MgdLib {
mgClass::mgClass()
{
m_pImpObj = new PureCClass();
}
mgClass::!mgClass()
{
delete m_pImpObj;
}
mgClass::~mgClass()
{
this->!mgClass();
}
String ^ mgClass::FirstName::get()
{
return gcnew String(m_pImpObj->getFirstName());
}
void mgClass::FirstName::set(String^ str)
{
pin_ptr<const WCHAR> wch = PtrToStringChars(str);
m_pImpObj->setFirstName(((std::wstring)wch).c_str());
}
String ^ mgClass::LastName::get()
{
return gcnew String(m_pImpObj->getLastName());
}
void mgClass::LastName::set(String^ str)
{
pin_ptr<const WCHAR> wch = PtrToStringChars(str);
m_pImpObj->setLastName(((std::wstring)wch).c_str());
}
String ^ mgClass::JoinName ()
{
return gcnew String(m_pImpObj->joinName());
}
}


然后,建立C#界面工程。C#工程reference引用托管工程生成的MgdLib.dll,并using命名空间MgdLib,就可以实现对托管包装类mgClass的调用了。(为了正确编译,需要调整好三个工程之间的依赖关系)



"Connect" Button 的单击函数里添加代码:

private void btnConnect_Click(object sender, EventArgs e)
{
mgClass myC = new mgClass();
myC.FirstName = txtFirstName.Text;
myC.LastName = txtLastName.Text;
String str = myC.JoinName();
txtFullName.Text = str;
}


最终的运行结果:



以上讲解了在C#代码中调用纯C++模块的基本流程。在本文章中我们再来实现一下C++代码调用C#代码的过程。我构造一个简单并且直观的例子:通过C++ UI 触发C# UI.

首先建立一个C#工程Class Library工程——CSharpUI



为该工程添加一个Form界面,并添加一个C#函数——InvokeUi()来构造并显示这个界面

namespace CSharpUI
{
public class Program
{
public static void InvokeUi()
{
//class Form1-----C# UI
Form1 fm = new Form1();
fm.ShowDialog();
}
}
}


建立一个托管的DYNAMIC LINK LIBRARY工程——MgdPro。

在MgdPro工程中用代码包装了对上述C#函数的调用,并将包装后的类从DLL导出。前提条件是reference CSharpUI.dll.

//.h file
#define DLLIMPEXP __declspec(dllexport)
class DLLIMPEXP MgdClass
{
public:
static void InvokeCsharpDlg();
};
//.cpp file
using namespace CSharpUI;
void MgdClass::InvokeCsharpDlg()
{
Program::InvokeUi();
}


最后,新建一个Dialog Based的C++工程——PureC++Pro(exe工程),在工程中静态链接MgdPro.dll.在Invoke按钮的触发函数里调用代码触发C#界面。void CPureCProDlg::OnBnClickedButton1()

{
// TODO: Add your control notification handler code here
//call managed c++ to invoke c# UI
MgdClass::InvokeCsharpDlg();
}


运行界面如下所示:

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: