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

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

2016-11-12 23:13 323 查看
关于C#和C++孰优孰劣的争论有很多,我不太喜欢这样的争论,感觉好像非要置对方于死地而后快。咱们伟大的毛主席说了:百花争放,百家齐鸣。

改革开放的总设计师邓小平也说了:黑猫白猫,逮住老鼠就是好猫。(呵呵,扯得太远了一点,愤青们千万不要砸砖头。)我想说的是:其实两种语言都各有自己的长处,取长补短才是硬道理。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;  
}  

最终的运行结果:



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