介绍
这是C/C++程序迷们经常谈论的一个话题同时也是一个复杂的难以理解的话题-指针!每次谈到C#大多数我遇到的人都持这样的观点-C#中没有指针的概念而实际上它已经被废除了取而代之的是C#中的非安全编程-如何在程序中使用指针不同于其字面意思的是使用指针编程并没有什么不安全的
它如此受关注的根本原因是非安全编程不同于习惯的NET开发规范而需要编程人员进行明确定本地环境设置(仅适用于本地执行)本文我将从区别两个最容易被疑惑的概念-非安全代码与非受控代码开始讨论非安全编程这个主题接下来我们将讨论如何编写非安全代码亦即如何在C#中使用指针
非安全还是非受控?
受控代码是指在CLR管理下执行的代码CLR负责了许多幕后的工作
管理对象的内存
进行类型验证
垃圾回收
说了这些实际就是要将用户从上述的这些工作中解脱出来了专心于业务实现用户不再需要直接手工地进行内存操作因为这些工作已由CLR完成了
另一方面非受控代码就是在CLR上下文外执行的代码了最好的例子就是我们平时使用的Win DLL比如kerneldlluserdll以及安装上我们系统上的各种COM组件如何为它们分配内存如何释放这些内存如何实现类型验证?这些工作都需要它们自己来完成一个典型的C++程序中分配一个字符指针的语句也是非受控代码的另一类例子因为作为一名编程者你要负责
调用内存分配函数
确保类型转换的结果正确
确保指针在使用完毕后其内存被释放
如果你留心上面的解释所有这些工作都是由CLR来完成以减轻编程者的负担
非安全代码是介于受控与非受控代码间的一种代码类型
非安全代码仍然象受控代码一样是在CLR的管理下执行的但在同时它又象非受控代码一样允许你通过指针直接访问内存因此你获得了两者的优点如果你正在编写写一个NET应用程序但同时又希望可以广泛使用Win DLL中的各种函数-需要使用指针的那么此时非安全代码就是你的救星了
我们已经明确了两者的区别后就开始编写实际的代码毫无疑问这才是最精彩的部分你还在想什么呢?
深入非安全代码
编写非安全代码需要使用特殊的关键字unsafe与fixed如果你还记得的话有三种指针操作符
*
&
>
任何使用了上述任一指针操作符的语句语句块或者函数都应用unsafe关键字标记为非安全代码就象这样
public unsafe void Triple(int *pInt)
{
*pInt=(*pInt)*;
}
上面这个函数只是将传入的参数的值扩大了两倍但是请注意传入的是这个参数的指针!因为这个函数使用了*操作符直接进行内存操作因此被标记为 unsafe
但是这里还是有一个问题回想一下上面的讨论非安全代码也是在CLR管理下的受控代码CLR可以自由地将对象移入内存中于是一个似是而非的原因可能导致内存洩漏这样做的结果是对于编程者可能在自觉不自觉中使这个变量的指针指向内存的其他地方
因此假设*pInt指向的地址是而CLR的内存重定位过程将会引发内存洩漏pInt之前指向在重定位后其指向的数据可能被存储在地址处于是大祸临头了!pInt指向的处存储的数据在经过重定位过程后无效了这也许就是NET很少提及指针的使用的原因吧你认为呢?
固定指针
在语句块前输入关键字fixed将会告诉CLR块内的对象不能重定位这样CLR就不会重定位指针指向的数据存储位置因此在C#中使用指针时使用关键字fixed将能阻止程序运行时无效指针的产生让我们看看它是如何工作的
using System;
class CData
{
public int x;
}
class CProgram
{
unsafe static void SetVal(int *pInt)
{
*pInt=;
}
public unsafe static void Main()
{
CData d = new CData();
ConsoleWriteLine(Previous value: {} dx);
fixed(int *p=&dx)
{
SetVal(p);
}
ConsoleWriteLine(New value: {} dx);
}
}
我们在这段代码里通过一个fixed块将CData对象数据成员(域)x的地址赋给了一个整数型指针p当fixed块中的语句被执行时这个指针p将一直指向原来的那块内存区域因为CLR已被指示暂时冻结这个变量直到该fixed块执行完毕一旦fixed块执行完毕这个对象就又能被CLR重新定位了
以上就是C#中使用指针编程的介绍关键是要说明语句块是unsafe并fixed的希望能因此提高你对C#中指针使用的知识!