100165348.htm
来自「C#高级编程(第三版),顶死你们。。 。up」· HTM 代码 · 共 142 行 · 第 1/3 页
HTM
142 行
<p class="MsoNormal" style="LINE-HEIGHT: 17pt"><span lang="EN-US" style="LETTER-SPACING: -0.1pt">C#</span><span style="FONT-FAMILY: 宋体; LETTER-SPACING: -0.1pt">引用主要用于使</span><span lang="EN-US" style="LETTER-SPACING: -0.1pt">C#</span><span style="FONT-FAMILY: 宋体; LETTER-SPACING: -0.1pt">语言易于使用,防止用户无意中执行某些破坏内存中内容的操作,另一方面,使用指针,就可以访问实际内存地址,执行新类型的操作。例如,可以给地址加上</span><span lang="EN-US" style="LETTER-SPACING: -0.1pt">4B</span><span style="FONT-FAMILY: 宋体; LETTER-SPACING: -0.1pt">,这样就可以查看甚至修改存储在新地址中的数据。</span></p>
<p class="MsoNormal" style="LINE-HEIGHT: 17pt"><span style="FONT-FAMILY: 宋体">下面是使用指针的两个主要原因:</span></p>
<p class="1" style="MARGIN-LEFT: 37.55pt; FTEL: -16.1pt"><span lang="EN-US">●<span style="FONT: 7pt 'Times New Roman'"> </span></span><span style="FONT-FAMILY: 宋体">向后兼容性。尽管</span><span lang="EN-US">.NET</span><span style="FONT-FAMILY: 宋体">运行库提供了许多工具,但仍可以调用旧的</span><span lang="EN-US">Windows API </span><span style="FONT-FAMILY: 宋体">函数。</span><span style="LETTER-SPACING: -1pt"> </span><span style="FONT-FAMILY: 宋体">对于某些操作来说,这可能是完成任务的惟一方式。这些</span><span lang="EN-US">API</span><span style="FONT-FAMILY: 宋体">函数都是用</span><span lang="EN-US">C</span><span style="FONT-FAMILY: 宋体">语言编写的,通常要求把指针作为其参数。但在许多情况下,还可以使用</span><span lang="EN-US">DllImport</span><span style="FONT-FAMILY: 宋体">声明,以避免使用指针,例如使用</span><span lang="EN-US">System.IntPtr</span><span style="FONT-FAMILY: 宋体">类。</span></p>
<p class="1" style="MARGIN-LEFT: 37.55pt; FTEL: -16.1pt"><span lang="EN-US">●<span style="FONT: 7pt 'Times New Roman'"> </span></span><span style="FONT-FAMILY: 宋体">性能。在一些情况下,速度是最重要的,而指针可以提供最优性能。假定用户知道自己在做什么,就可以确保以最高效的方式访问或处理数据。但是,注意在代码的其他区域中,不使用指针,也可以对性能做必要的改进。请使用代码配置文件,查找代码中的瓶颈,代码配置文件随</span><span lang="EN-US">VS.NET</span><span style="FONT-FAMILY: 宋体">一起安装。</span></p>
<p class="MsoNormal"><span style="FONT-FAMILY: 宋体">但是,这种低级内存访问也是有代价的。使用指针的语法比引用类型更复杂。而且,指针使用起来比较困难,需要非常高的编程技巧和很强的能力,仔细考虑代码所完成的逻辑操作,才能成功地使用指针。如果不仔细,使用指针很容易在程序中引入微妙的、难以查找的错误。例如很容易重写其他变量,导致堆栈溢出,访问某些没有存储变量的内存区域,甚至重写</span><span lang="EN-US">.NET</span><span style="FONT-FAMILY: 宋体">运行库所需要的代码信息,因而使程序崩溃。</span></p>
<p class="MsoNormal"><span style="FONT-FAMILY: 宋体">另外,如果使用指针就必须为代码获取代码访问安全机制的高级别信任,否则就不能执行。在默认的代码访问安全策略中,只有代码运行在本地机器上,这才是可能的。如果代码必须运行在远程地点,例如</span><span lang="EN-US">Internet</span><span style="FONT-FAMILY: 宋体">,用户就必须给代码授予额外的许可,代码才能工作。除非用户信任您和你的代码,否则他们不会授予这些许可,第</span><span lang="EN-US">14</span><span style="FONT-FAMILY: 宋体">章将讨论代码访问安全性。</span></p>
<p class="MsoNormal"><span style="FONT-FAMILY: 宋体">尽管有这些问题,但指针在编写高效的代码时是一种非常强大和灵活的工具,这里就介绍指针的使用。</span></p>
<p class="a3" style="MARGIN-TOP: 8.15pt; FTEL: 21.45pt"><span style="FONT-FAMILY: 黑体">注意:</span></p>
<p class="a1" style="FTEL: 21.45pt"><span style="FONT-FAMILY: 楷体_GB2312">这里强烈建议不要使用指针,因为如果使用指针,代码不仅难以编写和调试,而且无法通过</span><span lang="EN-US">CLR</span><span style="FONT-FAMILY: 楷体_GB2312">的内存类型安全检查</span><span lang="EN-US">(</span><span style="FONT-FAMILY: 楷体_GB2312">详见第</span><span lang="EN-US">1</span><span style="FONT-FAMILY: 楷体_GB2312">章</span><span lang="EN-US">)</span><span style="FONT-FAMILY: 楷体_GB2312">。</span></p>
<h4 style="FTEL: 21.45pt"><span lang="EN-US">1. </span><span style="FONT-FAMILY: 黑体">编写不安全的代码</span></h4>
<p class="MsoNormal"><span style="FONT-FAMILY: 宋体">因为使用指针会带来相关的风险,所以</span><span lang="EN-US">C#</span><span style="FONT-FAMILY: 宋体">只允许在特别标记的代码块中使用指针。标记代码所用的关键字是</span><span lang="EN-US">unsafe</span><span style="FONT-FAMILY: 宋体">。下面的代码把一个方法标记为</span><span lang="EN-US">unsafe</span><span style="FONT-FAMILY: 宋体">:</span></p>
<p class="2" style="MARGIN-TOP: 8.15pt; MARGIN-LEFT: 21.45pt; MARGIN-RIGHT: 0cm; FTEL: 18.45pt"><a ftel="unsafe2"><span lang="EN-US">unsafe int GetSomeNumber()</span></a></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; FTEL: 18.45pt"><span lang="EN-US">{</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; FTEL: 18.45pt"><span lang="EN-US"> // code that can use pointers</span></p>
<p class="2" style="MARGIN-TOP: 0cm; MARGIN-LEFT: 21.45pt; MARGIN-RIGHT: 0cm; FTEL: 18.45pt"><span lang="EN-US">}</span></p>
<p class="MsoNormal"><span style="FONT-FAMILY: 宋体">任何方法都可以标记为</span><span lang="EN-US">unsafe</span><span style="FONT-FAMILY: 宋体; LETTER-SPACING: -1pt">——</span><span style="LETTER-SPACING: -1pt"> </span><span style="FONT-FAMILY: 宋体">无论该方法是否应用了其他修饰符</span><span lang="EN-US">(</span><span style="FONT-FAMILY: 宋体">例如,静态方法、虚拟方法等</span><span lang="EN-US">)</span><span style="FONT-FAMILY: 宋体">。在这种方法中,</span><span lang="EN-US">unsafe</span><span style="FONT-FAMILY: 宋体">修饰符还会应用到方法的参数上,允许把指针用作参数。还可以把整个类或结构标记为</span><span lang="EN-US">unsafe</span><span style="FONT-FAMILY: 宋体">,表示所有的成员都是不安全的:</span></p>
<p class="2" style="MARGIN-TOP: 8.15pt; MARGIN-LEFT: 21.45pt; MARGIN-RIGHT: 0cm; FTEL: 18.45pt"><span lang="EN-US">unsafe class MyClass</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; FTEL: 18.45pt"><span lang="EN-US">{</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; FTEL: 18.45pt"><span lang="EN-US"> // any method in this class can now use pointers</span></p>
<p class="2" style="MARGIN-TOP: 0cm; MARGIN-LEFT: 21.45pt; MARGIN-RIGHT: 0cm; FTEL: 18.45pt"><span lang="EN-US">}</span></p>
<p class="MsoNormal"><span style="FONT-FAMILY: 宋体">同样,可以把成员标记为</span><span lang="EN-US">unsafe</span><span style="FONT-FAMILY: 宋体">:</span></p>
<p class="2" style="MARGIN-TOP: 8.15pt; MARGIN-LEFT: 21.45pt; MARGIN-RIGHT: 0cm; FTEL: 18.45pt"><span lang="EN-US">class MyClass</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; FTEL: 18.45pt"><span lang="EN-US">{</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; FTEL: 18.45pt"><span lang="EN-US"> unsafe int *pX; // declaration of a pointer field in a class</span></p>
<p class="2" style="MARGIN-TOP: 0cm; MARGIN-LEFT: 21.45pt; MARGIN-RIGHT: 0cm; FTEL: 18.45pt"><span lang="EN-US">}</span></p>
<p class="MsoNormal"><span style="FONT-FAMILY: 宋体">也可以把方法中的一个代码块标记为</span><span lang="EN-US">unsafe</span><span style="FONT-FAMILY: 宋体">:</span></p>
<p class="2" style="MARGIN-TOP: 8.15pt; MARGIN-LEFT: 21.45pt; MARGIN-RIGHT: 0cm; FTEL: 18.45pt"><span lang="EN-US">void MyMethod()</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; FTEL: 18.45pt"><span lang="EN-US">{</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; FTEL: 18.45pt"><span lang="EN-US"> // code that doesn't use pointers</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; FTEL: 18.45pt"><span lang="EN-US"> unsafe</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; FTEL: 18.45pt"><span lang="EN-US"> {</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; FTEL: 18.45pt"><span lang="EN-US"> // unsafe code that uses pointers here</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; FTEL: 18.45pt"><span lang="EN-US"> }</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; FTEL: 18.45pt"><span lang="EN-US"> // more 'safe' code that doesn't use pointers</span></p>
<p class="2" style="MARGIN-TOP: 0cm; MARGIN-LEFT: 21.45pt; MARGIN-RIGHT: 0cm; FTEL: 18.45pt"><span lang="EN-US">}</span></p>
<p class="MsoNormal"><span style="FONT-FAMILY: 宋体">但要注意,不能把局部变量本身标记为</span><span lang="EN-US">unsafe</span><span style="FONT-FAMILY: 宋体">:</span></p>
<p class="2" style="MARGIN-TOP: 8.15pt; MARGIN-LEFT: 21.45pt; MARGIN-RIGHT: 0cm; FTEL: 18.45pt"><span lang="EN-US">int MyMethod()</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; FTEL: 18.45pt"><span lang="EN-US">{ </span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; FTEL: 18.45pt"><span lang="EN-US"> unsafe int *pX; // WRONG</span></p>
<p class="2" style="MARGIN-TOP: 0cm; MARGIN-LEFT: 21.45pt; MARGIN-RIGHT: 0cm; FTEL: 18.45pt"><span lang="EN-US">}</span></p>
<p class="MsoNormal"><span style="FONT-FAMILY: 宋体">如果要使用不安全的局部变量,就需要在方法或不安全的语句块中声明和使用它。在使用指针前还有一步要完成。</span><span lang="EN-US">C#</span><span style="FONT-FAMILY: 宋体">编译器会拒绝不安全的代码,除非告诉编译器代码包含不安全的代码块。标记所用的关键字是</span><span lang="EN-US">unsafe</span><span style="FONT-FAMILY: 宋体">。因此,要编译包含不安全代码的文件</span><span lang="EN-US">MySource.cs(</span><span style="FONT-FAMILY: 宋体">假定没有其他编译器选项</span><span lang="EN-US">)</span><span style="FONT-FAMILY: 宋体">,就要使用下述命令:</span></p>
<p class="a6" style="MARGIN-TOP: 8.15pt; MARGIN-LEFT: 21.45pt; MARGIN-RIGHT: 0cm; FTEL: 18.45pt"><strong><span lang="EN-US">csc /unsafe MySource.cs</span></strong></p>
<p class="MsoNormal"><span style="FONT-FAMILY: 宋体">或者</span></p>
<p class="a6" style="MARGIN-TOP: 8.15pt; MARGIN-LEFT: 21.45pt; MARGIN-RIGHT: 0cm; FTEL: 18.45pt"><strong><span lang="EN-US">csc –unsafe MySource.cs</span></strong></p>
<p class="a3" style="MARGIN-TOP: 8.15pt; FTEL: 21.45pt"><span style="FONT-FAMILY: 黑体">注意:</span></p>
<p class="a1" style="FTEL: 21.45pt"><span style="FONT-FAMILY: 楷体_GB2312">如果使用</span><span lang="EN-US">Visual Studio .NET</span><span style="FONT-FAMILY: 楷体_GB2312">,就可以在项目属性中找到编译不安全代码的选项。对于本节中可下载示例的</span><span lang="EN-US">Visual Studio .NET</span><span style="FONT-FAMILY: 楷体_GB2312">版本,我们已经设置了不安全编译选项。</span></p>
<h4 style="FTEL: 21.45pt"><span lang="EN-US">2. </span><span style="FONT-FAMILY: 黑体">指针的语法</span></h4>
<p class="MsoNormal"><span style="FONT-FAMILY: 宋体">把代码块标记为</span><span lang="EN-US">unsafe</span><span style="FONT-FAMILY: 宋体">后,就可以使用下面的语法声明指针:</span></p>
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?