在C#中使用C++编写的类——用托管C++进行封装
现在在Windows下的应用程序开发,VS.Net占据了绝大多数的份额。因此很多以前搞VC++开发的人都转向用更强大的VS.Net。在这种情况下,有很多开发人员就面临了如何在C#中使用C++开发好的类的问题。下面就用一个完整的实例来详细说明怎样用托管C++封装一个C++类以提供给C#使用。
比如,现在有一个工程名为NativeCppDll的由C++编写的DLL,里面输出了一个CPerson类。下面是具体的代码:
- // NativeCppDll.h
- #pragma once
- #ifndef LX_DLL_CLASS_EXPORTS
- #define LX_DLL_CLASS __declspec(dllexport)
- #else
- #define LX_DLL_CLASS __declspec(dllimport)
- #endif
- class LX_DLL_CLASS CPerson
- {
- public:
- CPerson();
- CPerson(const wchar_t *pName, const wchar_t cSex, int iAge);
- void SetName(const wchar_t *pName);
- wchar_t * GetName();
- void SetSex(const wchar_t cSex);
- wchar_t GetSex();
- void SetAge(int iAge);
- int GetAge();
- wchar_t * GetLastError();
- private:
- wchar_t m_szName[128];
- wchar_t m_cSex;
- int m_iAge;
- wchar_t m_szLastError[128];
- void ShowError();
- };
- // NativeCppDll.cpp
- #include "stdafx.h"
- #include "NativeCppDll.h"
- #include
- #include
- using namespace std;
- CPerson::CPerson()
- {
- wcscpy_s(m_szName, _T("No Name"));
- m_cSex = 'N';
- m_iAge = 0;
- wcscpy_s(m_szLastError, _T("No Error"));
- }
- CPerson::CPerson(const wchar_t *pName, const wchar_t cSex, int iAge)
- {
- wcscpy_s(m_szLastError, _T("No Error"));
- SetName(pName);
- SetSex(cSex);
- SetAge(iAge);
- }
- void CPerson::SetName(const wchar_t *pName)
- {
- if ((pName == NULL) || (wcslen(pName) == 0) || (wcslen(pName) > 127))
- {
- wcscpy_s(m_szName, _T("No Name"));
- wcscpy_s(m_szLastError, _T("The length of the input name is out of range."));
- ShowError();
- return;
- }
- wcscpy_s(m_szName, pName);
- }
- wchar_t * CPerson::GetName()
- {
- return m_szName;
- }
- void CPerson::SetSex(const wchar_t cSex)
- {
- if ((cSex != 'F') && (cSex != 'M') && (cSex != 'm') && (cSex != 'f'))
- {
- m_cSex = 'N';
- wcscpy_s(m_szLastError, _T("The input sex is out of [F/M]."));
- ShowError();
- return;
- }
- m_cSex = cSex;
- }
- wchar_t CPerson::GetSex()
- {
- return m_cSex;
- }
- void CPerson::SetAge(int iAge)
- {
- if ((iAge < 0) || (iAge > 150))
- {
- m_iAge = 0;
- wcscpy_s(m_szLastError, _T("The input age is out of range."));
- ShowError();
- return;
- }
- m_iAge = iAge;
- }
- int CPerson::GetAge()
- {
- return m_iAge;
- }
- wchar_t * CPerson::GetLastError()
- {
- return m_szLastError;
- }
- void CPerson::ShowError()
- {
- cerr << m_szLastError << endl;
- }
这是一个很典型的由C++开发的DLL,输出一个完整的C++类。如果现在要求开发一个C#工程,需要用到这个DLL中输出的C++类CPerson,该怎么办呢?针对这个例子来说,类CPerson非常小,可以用C#重新写一个跟这个C++类一样的类。可是,如果需要的C++类很大,或者很多的时候,重写工程将非常庞大。而且这样没有对现有的代码进行重用,浪费了现有资源,开发起来费时费力。
当然,还是有方法解决这个问题的。那就是用托管C++将C++类给封装一下,然后再提供给C#来使用。下面就用代码来详细说明怎样用托管C++来封装上面的那个C++类。
首先,要创建一个托管C++的DLL工程ManageCppDll,然后在里面添加下面的代码:
- // ManageCppDll.h
- #pragma once
- #define LX_DLL_CLASS_EXPORTS
- #include "../NativeCppDll/NativeCppDll.h"
- using namespace System;
- namespace ManageCppDll
- {
- public ref class Person
- {
- // 包装所有类CPerson的公有成员函数
- public:
- Person();
- Person(String ^ strName, Char cSex, int iAge);
- ~Person();
- property String ^ Name
- {
- void set(String ^ strName);
- String ^ get();
- }
- property Char Sex
- {
- void set(Char cSex);
- Char get();
- }
- property int Age
- {
- void set(int iAge);
- int get();
- }
- String ^ GetLastError();
- private:
- // 类CPerson的指针,用来调用类CPerson的成员函数
- CPerson *m_pImp;
- };
- };
从这个头文件就能看出来,这是对C++类CPerson的包装。类Person的所有公有成员函数都跟C++类CPerson一样,只不过成员函数的参数和返回值就改成了托管C++的类型,这也是让类Person能在C#中使用的首要条件。当然只需要对公有成员函数进行封装,对于保护成员函数和私有成员函数则不必做任何封装。
类Person仅有一个私有的成员变量:一个类CPerson的指针。而类Person的所有成员函数的实现都是靠这个CPerson指针来调用类CPerson的相应成员函数来实现。
下面是具体的实现代码:
- // ManageCppDll.cpp
- #include "stdafx.h"
- #include "ManageCppDll.h"
- #include
- namespace ManageCppDll
- {
- // 在构造函数中创建类CPerson的对象并在析构函数中将该对象销毁
- // 所有的成员函数实现都是通过指针m_pImp调用类CPerson的相应成员函数实现
- Person::Person()
- {
- m_pImp = new CPerson();
- }
- Person::Person(String ^ strName, Char cSex, int iAge)
- {
- // 将string转换成C++能识别的指针
- pin_ptr<</SPAN>const wchar_t> wcName = PtrToStringChars(strName);
- m_pImp = new CPerson(wcName, cSex, iAge);
- }
- Person::~Person()
- {
- // 在析构函数中删除CPerson对象
- delete m_pImp;
- }
- void Person::Name::set(String ^ strName)
- {
- pin_ptr<</SPAN>const wchar_t> wcName = PtrToStringChars(strName);
- m_pImp->SetName(wcName);
- }
- String ^ Person::Name::get()
- {
- return gcnew String(m_pImp->GetName());
- }
- void Person::Sex::set(Char cSex)
- {
- m_pImp->SetSex(cSex);
- }
- Char Person::Sex::get()
- {
- return m_pImp->GetSex();
- }
- void Person::Age::set(int iAge)
- {
- m_pImp->SetAge(iAge);
- }
- int Person::Age::get()
- {
- return m_pImp->GetAge();
- }
- String ^ Person::GetLastError()
- {
- return gcnew String(m_pImp->GetLastError());
- }
- };
如果要在C#中使用类Person,首先要添加对ManageCppDll.dll的引用,然后就可以像用普通的C#类一样的使用类Person了。比如下面这样的代码:
- using ManageCppDll;
- Person person = new Person();
- person.Name = "StarLee";
- person.Sex = 'M';
- person.Age = 28;
熟悉设计模式的看了上面的代码肯定会发现,这样的设计跟BRIDGE模式如出一辙。其实,上面的方法也算是一种BRIDGE模式,由托管C++充当了C#中使用用C++开发的类的桥梁。另外,这种形式也可以理解为ADAPTER模式,托管C++类Person就是C++类CPerson的一个适配器。通过这个桥梁,可以很容易的重用以前用C++开发的类,让这些C++类继续在C#中发挥它们的效用,让开发变得事半功倍。
博客来源:http://blog.csdn.net/starlee/article/details/2864588
在C#中使用C++编写的类——用托管C++进行封装的更多相关文章
- 在C#中使用C++编写的类——用托管C++进行封装[转]
现在在Windows下的应用程序开发,VS.Net占据了绝大多数的份额.因此很多以前搞VC++开发的人都转向用更强大的VS.Net.在这种情况 下,有很多开发人员就面临了如何在C#中使用C++开发好的 ...
- 在C++中使用C#编写的类2
在那篇<在C#中使用C++编写的类>中我介绍了如何在C#中使用C++编写的类.可是由于C#在用户界面设计.数据库存储和XML文件读取等方面的优势,有时候也会出现要在C++中使用C#编写的类 ...
- 在C#中使用C++编写的类
现在在Windows下的应用程序开发,VS.Net占据了绝大多数的份额.因此很多以前搞VC++开发的人都转向用更强大的VS.Net.在这种情况下,有很多开发人员就面临了如何在C#中使用C++开发好的类 ...
- 在C#中使用C++编写的类1
转载地址:http://blog.csdn.net/starlee/article/details/2864588 现在在Windows下的应用程序开发,VS.Net占据了绝大多数的份额.因此很多以前 ...
- 22.编写一个类A,该类创建的对象可以调用方法showA输出小写的英文字母表。然后再编写一个A类的子类B,子类B创建的对象不仅可以调用方法showA输出小写的英文字母表,而且可以调用子类新增的方法showB输出大写的英文字母表。最后编写主类C,在主类的main方法 中测试类A与类B。
22.编写一个类A,该类创建的对象可以调用方法showA输出小写的英文字母表.然后再编写一个A类的子类B,子类B创建的对象不仅可以调用方法showA输出小写的英文字母表,而且可以调用子类新增的方法sh ...
- 编写高质量代码改善C#程序的157个建议——建议112:将现实世界中的对象抽象为类,将可复用对象圈起来就是命名空间
建议112:将现实世界中的对象抽象为类,将可复用对象圈起来就是命名空间 在我们身边的世界中,对象是什么?对象就是事物,俗称“东西”.那么,什么东西算得上是一个对象呢?对象有属性.有行为.以动物为例,比 ...
- 题目一:编写一个类Computer,类中含有一个求n的阶乘的方法
作业:编写一个类Computer,类中含有一个求n的阶乘的方法.将该类打包,并在另一包中的Java文件App.java中引入包,在主类中定义Computer类的对象,调用求n的阶乘的方法(n值由参数决 ...
- 35.按要求编写Java程序: (1)编写一个接口:InterfaceA,只含有一个方法int method(int n); (2)编写一个类:ClassA来实现接口InterfaceA,实现int method(int n)接口方 法时,要求计算1到n的和; (3)编写另一个类:ClassB来实现接口InterfaceA,实现int method(int n)接口 方法时,要求计算n的阶乘(n
35.按要求编写Java程序: (1)编写一个接口:InterfaceA,只含有一个方法int method(int n): (2)编写一个类:ClassA来实现接口InterfaceA,实现in ...
- C++中public,protected,private派生类继承问题和访问权限问题
C++中public,protected,private派生类继承问题和访问权限问题 当一个子类从父类继承时,父类的所有成员成为子类的成员,此时对父类成员的访问状态由继承时使用的继承限定符决定. 1. ...
随机推荐
- SLAM基础算法(1):卡尔曼滤波
对于一个正在运动中的小车来说,如何准确的知道它所处的位置? 理论家说:我可以通过牛顿公式来计算! 实践家说:给它装个GPS不就得了! 好吧,你们说的听上去都很有道理,但我们到底该相信谁? 现实情况是: ...
- HelloWord!
HelloWorld 1.新建一个文件夹,存放代码 2.新建一个java文件后缀为.java Hello.java (注意后缀 如系统没打开显示后缀需要打开) 3.编写HelloWorld代码 (建议 ...
- Spark—local模式环境搭建
Spark--local模式环境搭建 一.Spark运行模式介绍 1.本地模式(loca模式):spark单机运行,一般用户测试和开发使用 2.Standalone模式:构建一个主从结构(Master ...
- Python+Requests+Bs4(解析)爬取某诗词信息(数据分析二)
1.环境安装 - 需要将pip源设置为国内源,阿里源.豆瓣源.网易源等 - windows (1)打开文件资源管理器(文件夹地址栏中) (2)地址栏上面输入 %appdata% (3)在这里面新建一个 ...
- 7Java基础补充
1.标准Java bean写法 包括:private修饰的成员变量.getter和setter以及无参和有多个参数的有参构造方法 2.String原理 String底层是字节数组byte[]. Str ...
- 【爬虫系列】0. 无内鬼,破解前端JS参数签名
PS:这是一个系列,坐等我慢慢填坑. PS:不太会直接能跑的代码,抛砖引玉. PS:那些我也不太熟练的就不搞了,包括(破滑块.验证码..) PS: 反编译搞Apk会有很长的几个文章,稍后慢慢更. 最近 ...
- PAT乙级:1077 互评成绩计算 (20分)
PAT乙级:1077 互评成绩计算 (20分) 在浙大的计算机专业课中,经常有互评分组报告这个环节.一个组上台介绍自己的工作,其他组在台下为其表现评分.最后这个组的互评成绩是这样计算的:所有其他组的评 ...
- 第十一篇 -- QMainWindow与QAction(斜体-粗体-下划线)
效果图: ui_mainWindow.py # -*- coding: utf-8 -*- # Form implementation generated from reading ui file ' ...
- python基础之读取xml
python怎么操作xml文件详细介绍链接:https://www.jb51.net/article/50812.htm 从结构上来说,xml很像常见的HTML超文本标记语言.不过超文本语言被设计用来 ...
- 离线安装rpm包并解决依赖(升级vsftpd为例)
背景 实际开发中,我们的linux服务器是处理离线状态的,并不能访问互联网.如果此时要在linux上安装或者升级软件,就只能通过rpm包的安装方式.rpm包安装有一个缺陷,就是不能处理安装包的依赖问 ...