在C++中使用C#编写的类2
在那篇《在C#中使用C++编写的类》中我介绍了如何在C#中使用C++编写的类。可是由于C#在用户界面设计、数据库存储和XML文件读取等方面的优势,有时候也会出现要在C++中使用C#编写的类的情况。下面就用一个完整的实例来说明怎样在C++中使用C#编写的类。
比如说,现在有一个用C#编写的DLL工程CsharpDll里面有一个Person类:
- // Person.cs
- using System;
- namespace CsharpDll
- {
- public class Person
- {
- public Person()
- {
- Name = "No Name";
- Sex = 'N';
- Age = 0;
- m_strLastError = "No Error";
- }
- public Person(string strName, char cSex, int iAge)
- {
- m_strLastError = "No Error";
- Name = strName;
- Sex = cSex;
- Age = iAge;
- }
- public string Name
- {
- get
- {
- return m_strName;
- }
- set
- {
- if ((String.IsNullOrEmpty(value)) || (value.Length > 127))
- {
- m_strName = "No Name";
- m_strLastError = "The length of the input name is out of range.";
- return;
- }
- m_strName = value;
- }
- }
- public char Sex
- {
- get
- {
- return m_cSex;
- }
- set
- {
- if ((value != 'F') && (value != 'M') && (value != 'm') && (value != 'f'))
- {
- m_cSex = 'N';
- m_strLastError = "The input sex is out of [F/M].";
- return;
- }
- m_cSex = value;
- }
- }
- public int Age
- {
- get
- {
- return m_iAge;
- }
- set
- {
- if ((value < 0) || (value > 150))
- {
- m_iAge = 0;
- m_strLastError = "The input age is out of range.";
- return;
- }
- m_iAge = value;
- }
- }
- public string LastError
- {
- get
- {
- return m_strLastError;
- }
- }
- private string m_strName;
- private char m_cSex;
- private int m_iAge;
- private string m_strLastError;
- }
- }
如果需要在C++中使用这个C#编写的Person类,就需要用托管C++来对这个C#进行包装,将它包装成一个C++能用的类。
首先,要创建一个托管C++的DLL工程ManageCppDll。并且,要添加对CsharpDll.dll的引用。然后对C#类所有的公有属性和方法进行包装。下面是具体的代码:
- // ManageCppDll.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);
- ~CPerson();
- 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:
- // 用一个void指针指向Person的对象
- // 所有公有成员函数的实现都是通过这个对象来实现
- void *m_pImp;
- wchar_t m_szName[128];
- wchar_t m_szLastError[128];
- };
- // ManageCppDll.cpp
- #include "stdafx.h"
- #include "ManageCppDll.h"
- #include <vcclr.h>
- #include <string.h>
- #include <stdlib.h>
- using namespace System;
- using namespace System::Runtime::InteropServices;
- using namespace CsharpDll;
- // 将GCHandle转换成为void指针
- #define __GCHANDLE_TO_VOIDPTR(x) ((GCHandle::operator System::IntPtr(x)).ToPointer())
- // 将void指针转换为GCHandle
- #define __VOIDPTR_TO_GCHANDLE(x) (GCHandle::operator GCHandle(System::IntPtr(x)))
- // 辅助函数
- // 将void指针指向的对象转换成为Person对象
- inline Person ^ GetImpObj(void *pHandle)
- {
- Person ^ person = nullptr;
- if (pHandle != NULL)
- {
- person = static_cast<Person^>(__VOIDPTR_TO_GCHANDLE(pHandle).Target);
- }
- return person;
- }
- CPerson::CPerson()
- {
- m_pImp = NULL;
- Person ^ person = gcnew Person();
- // 创建GCHandle并将它转换成void指针保存到成员变量中
- GCHandle handle = GCHandle::Alloc(person);
- m_pImp = __GCHANDLE_TO_VOIDPTR(handle);
- }
- CPerson::CPerson(const wchar_t *pName, const wchar_t cSex, int iAge)
- {
- m_pImp = NULL;
- Person ^ person = gcnew Person();
- person->Name = gcnew String(pName);
- person->Sex = cSex;
- person->Age = iAge;
- GCHandle handle = GCHandle::Alloc(person);
- m_pImp = __GCHANDLE_TO_VOIDPTR(handle);
- }
- CPerson::~CPerson()
- {
- if (m_pImp == NULL)
- return;
- // 释放GCHandle
- GCHandle handle = __VOIDPTR_TO_GCHANDLE(m_pImp);
- handle.Free();
- m_pImp = NULL;
- }
- void CPerson::SetName(const wchar_t *pName)
- {
- // 将void指针转换成Person指针
- // 并用该指针调用相应的公有属性或方法
- Person ^ person = GetImpObj(m_pImp);
- person->Name = gcnew String(pName);
- }
- wchar_t * CPerson::GetName()
- {
- Person ^ person = GetImpObj(m_pImp);
- // 将C#返回的字符串转换为wchat_t*指针能指向的地址
- wchar_t * pName = static_cast<wchar_t*>(Marshal::StringToHGlobalUni(person->Name).ToPointer());
- wcscpy_s(m_szName, pName);
- Marshal::FreeHGlobal(System::IntPtr(pName)); // 释放内存
- return m_szName;
- }
- void CPerson::SetSex(const wchar_t cSex)
- {
- Person ^ person = GetImpObj(m_pImp);
- person->Sex = cSex;
- }
- wchar_t CPerson::GetSex()
- {
- Person ^ person = GetImpObj(m_pImp);
- return person->Sex;
- }
- void CPerson::SetAge(int iAge)
- {
- Person ^ person = GetImpObj(m_pImp);
- person->Age = iAge;
- }
- int CPerson::GetAge()
- {
- Person ^ person = GetImpObj(m_pImp);
- return person->Age;
- }
- wchar_t * CPerson::GetLastError()
- {
- Person ^ person = GetImpObj(m_pImp);
- wchar_t * pLastError = static_cast<wchar_t*>(Marshal::StringToHGlobalUni(person->LastError).ToPointer());
- wcscpy_s(m_szLastError, pLastError);
- Marshal::FreeHGlobal(System::IntPtr(pLastError));
- return m_szLastError;
- }
现在对上面代码中所用到的一些相关背景知识进行一下介绍。
GCHandle结构提供从非托管内存访问托管对象的方法。
GCHandle.Alloc方法(Object)为指定的对象分配Normal句柄。它保护对象不被垃圾回收。当不再需要GCHandle时,必须通过Free将其释放。Normal句柄类型表示不透明句柄,这意味着无法通过此句柄解析固定对象的地址。可以使用此类型跟踪对象,并防止它被垃圾回收器回收。当非托管客户端持有对托管对象的唯一引用(从垃圾回收器检测不到该引用)时,此枚举成员很有用。
上面的代码中,在类CPerson的构造函数中用GCHandle为C#类Person的对象分配一个句柄,并将该句柄转换为void指针存放在成员变量中,以保证这个对象不会被垃圾回收器回收。然后在类CPerson的析构函数中释放这个句柄,将C#类Person的对象的回收权交给系统。
Marshal类提供了一个方法集,这些方法用于分配非托管内存、复制非托管内存块、将托管类型转换为非托管类型,此外还提供了在与非托管代码交互时使用的其他杂项方法。
Marshal..::.StringToHGlobalUni方法向非托管内存复制托管String的内容。StringToHGlobalUni对于自定义封送处理或者在混合托管和非托管代码时很有用。由于该方法分配字符串所需的非托管内存,因此应始终通过调用FreeHGlobal释放内存。
(更多关于上面介绍的背景知识可以搜索MSDN。说实在的MSDN真是一个宝库!VS能在Windows平台开发中取得绝大多数的份额,除了因为它和Windows都是微软开发的之外,MSDN的完备性功不可没!)
通过上面的方法,就把一个C#编写的类Person用托管C++给封装成了一个C++可以使用的CPerson类。我们可以在C++的工程中像使用一般的C++类一样使用类CPerson。比如下面的代码。
- CPerson person(_T("StarLee"), 'M', 28);
- person.SetName(_T("StarLee"));
- person.SetSex('M');
- person.SetAge(28);
- wcout << "Name: " << person.GetName() << " Sex: " << person.GetSex() << " Age: " << person.GetAge() << endl;
- wcout << "Error: " << person.GetLastError() << endl;
这里的方法跟《在C#中使用C++编写的类》一样,都是借用托管C++这个桥梁来沟通C++编写的类和C#编写的类,使在C++中使用C#编写的类和在C#中使用C++编写的类成为现实。
在C++中使用C#编写的类2的更多相关文章
- 在C#中使用C++编写的类
现在在Windows下的应用程序开发,VS.Net占据了绝大多数的份额.因此很多以前搞VC++开发的人都转向用更强大的VS.Net.在这种情况下,有很多开发人员就面临了如何在C#中使用C++开发好的类 ...
- 在C#中使用C++编写的类1
转载地址:http://blog.csdn.net/starlee/article/details/2864588 现在在Windows下的应用程序开发,VS.Net占据了绝大多数的份额.因此很多以前 ...
- 在C#中使用C++编写的类——用托管C++进行封装[转]
现在在Windows下的应用程序开发,VS.Net占据了绝大多数的份额.因此很多以前搞VC++开发的人都转向用更强大的VS.Net.在这种情况 下,有很多开发人员就面临了如何在C#中使用C++开发好的 ...
- 在C#中使用C++编写的类——用托管C++进行封装
现在在Windows下的应用程序开发,VS.Net占据了绝大多数的份额.因此很多以前搞VC++开发的人都转向用更强大的VS.Net.在这种情况下,有很多开发人员就面临了如何在C#中使用C++开发好的类 ...
- 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. ...
随机推荐
- C++类成员常量
由于#define 定义的宏常量是全局的,不能达到目的,于是想当然地觉得应该用const 修饰数据成员来实现.const 数据成员的确是存在的,但其含义却不是我们所期望的.const 数据成员只在某个 ...
- 编写可维护的JavaScript—语句和表达式&变量、函数和运算符
语句和表达式 所有的块语句都应当使用花括号.包括: if for while do…while… try…catch…finally //不好的写法 if (condition) doSomethin ...
- database schema
数据中有4个Schema无法被删除 ● dbo, 具有db_owner或者db_ddl_admin 的用户,新创建对象默认schema就是dbo ● guest , 用来给guest 用户使用,这个s ...
- java多线程——同步块synchronized详解
Java 同步块(synchronized block)用来标记方法或者代码块是同步的.Java同步块用来避免竞争.本文介绍以下内容: Java同步关键字(synchronzied) 实例方法同步 静 ...
- 纯css画哆啦A梦
今天有点无聊,照着网上的图写了个哆啦A梦,无技术可言,纯考耐心. <!doctype html> <html lang="en"> <head> ...
- HTML5 DTD
HTML5/HTML 4.01/XHTML 元素和有效的 DTD 下面的表格列出了所有的 HTML5/HTML 4.01/XHTML 元素,以及它们会出现在什么文档类型 (DTD) 中: 标签 HTM ...
- 第五节 Code 128 码
128码开始於1981年推出,是一种长度可变.连续性的字母数字条码.与其他一维条码比较起来,128码是较为复杂的条码系统,而其所能支援的字元也相对地比其他一维条码来得多,又有不同的编码方式可供交互运用 ...
- Android EditText 无法换行
问题 关于控制是否换行的属性android:singleLine 当值为true的时候,只能一行,不换行 当值为false的时候,可以换行 但是现在遇到一个问题: <EditText andro ...
- Codeforces Round #198 (Div. 2) 340C
C. Tourist Problem time limit per test 1 second memory limit per test 256 megabytes input standard i ...
- IOS算法(三)之插入排序
直接插入排序(Insertion Sort)的基本思想是:每次将一个待排序的记录,按其keyword大小插入到前面已经排好序的子序列中的适当位置,直到所有记录插入完毕为止. 设数组为a[0-n-1]. ...