關於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;
- }
最終的運行結果:
編輯推薦】