互操作系列文章:
-
.NET简谈互操作(一:开篇介绍)
-
.NET简谈互操作(二:先睹为快)
-
.NET简谈互操作(三:基础知识之DllImport特性)
-
.NET简谈互操作(四:基础知识之Dispose非托管内存)
-
.NET简谈互操作(五:基础知识之Dynamic平台调用)
-
.NET简谈互操作(六:基础知识之提升平台调用性能)
-
.NET简谈互操作(七:数据封送之介绍)
我们继续.NET互操作学习。互操作的基础知识已经差不多完了,当然一篇小小的文章很难全面的讲述互操作的方方面面,本人只是总结出关键的地方好让我们能入个门,在后期如果想要更深入的学习,肯定需要一本详细而全面的书籍才行。想要精通.NET互操作当然也少不了对非托管的技术学习,C++、COM等等,只有既熟悉.NET也熟悉非托管技术才能将互操作融会贯通。从这篇文章起我们将进入到.NET互操作的数据封送阶段,数据封送是.NET/Pinvoke关键的部分,任何托管代码想要和非托管代码互操作,少不了数据的传递返回;[王清培版权所有,转载请给出署名]
1:
在托管代码调用非托管代码的时候,数据经历了很复杂的封送。由于托管的数据类型与非托管的数据类型内存结构可能是不一样的,要想将托管数据参数传递到非托管代码中,并且能成功的接受到非托管的返回值,我们需要很严格的按照双方的数据类型约定来才行。比如在C++中的Char*是一个字符指针,当我们想要将字符串传递到非托管代码中时,我们需要考虑怎么将参数无差错的封送到非托管代码。而在托管C#中的String类型是一个引用类型,两者有相同点,也有不同点。
在我们用.NET/PInvoke进行Win32API的调用的时候,大多数的情况下我们是需要传递某种结构类型给API,然后在接受返回值;在数据封送的过程中,有些概念是我们平时不曾碰见的。.NET数据封送很智能,CLR的封送拆收器能通过识别我们传递的数据类型情况进行自动选择封送数据的方式,比如我们将一个Class的类类型传递给非托管API,默认的Class类是不能进行互操作使用的,必须加上相应的特性进行标记,编译器编译的时候能识别出这是要进行封送的数据类型;将Class类型进行封送时,封送拆收器会进行判断,如果我们传递给非托管代码的Class中的所有内部对象都是平台数据类型,那么CLR会将这个对象在内存中锁定,然后直接将内存地址封送给非托管代码,非托管代码直接对这数据进行操作。这是封送引用地址的方式,如果我们传递给非托管API的是非平台类型,那么CLR的封送拆收收器会将我们的托管类型复制出来进行非托管类型转换,然后将转换后的数据传递给非托管,这样的过程是复制数据的过程。互操作的数据封送基本上就是这两种,1数据的复制封送,2数据的内存地址封送;
下面我们用一副图来表达我上面所说的原理。
2:
如果托管的数据类型与非托管的数据类型在内存中是等价的,那么CLR进行封送的方式会很简单。如果托管的数据类型与非托管的数据类型是不等价的,那么CLR会进行相应复制转换操作,当然这样会丢失数据内存泄漏都是有可能的,但是我们不用怕,.NET为我们做好了很好的互操作桥梁,我们只要对要封送的数据进行一系列的设置就能很成功的进行数据封送了;
总结:这篇文章主要给大家介绍一下,关于托管与非托管的数据封送相关的概念,下面我们将学习互操作的数据封送;