100165597.htm

来自「C#高级编程(第三版),顶死你们。。 。up」· HTM 代码 · 共 138 行 · 第 1/2 页

HTM
138
字号
<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">finally</span><span style="FONT-FAMILY: 宋体">块中,可以释放已经使用的任何资源。这种方式的惟一麻烦是必须确保关闭连接。很容易忘记在</span><span lang="EN-US">finally</span><span style="FONT-FAMILY: 宋体">块中添加关闭连接的命令,所以应在编码风格上添加一些不容易出现反常情况的内容。</span></p>
<p class="MsoNormal"><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">try<sup>&hellip;</sup>catch<sup>&hellip;</sup>finally</span><span style="FONT-FAMILY: 宋体">块的层次有时可能不容易看懂。但还有另一个方式可以确保资源的关闭<span style="LETTER-SPACING: -1.2pt">&mdash;&mdash;</span></span><span style="LETTER-SPACING: -1.2pt"> </span><span style="FONT-FAMILY: 宋体">使用</span><span lang="EN-US">using</span><span style="FONT-FAMILY: 宋体">语句。</span></p>
<h4 style="FTEL: 21.45pt"><a ftel="_Toc536518867"><span lang="EN-US">2. </span></a><span style="FONT-FAMILY: 黑体">第二种方式<span style="LETTER-SPACING: -1pt">&mdash;&mdash;</span></span><span style="LETTER-SPACING: -1pt"> </span><span style="FONT-FAMILY: 黑体">使用</span><span lang="EN-US">using</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">.NET</span><span style="FONT-FAMILY: 宋体">在对象不再引用之后清理它们的方法是使用非决定性的析构方式,这已经引起了一个非常热烈的讨论。</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">C++</span><span style="FONT-FAMILY: 宋体">析构函数。所以,如果出现了异常,但没有捕获,有析构函数的所有对象就会调用它们的析构函数。</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">.NET</span><span style="FONT-FAMILY: 宋体">可执行程序的各种问题。幸运的是,我们还有解决的方法。下面的代码说明了如何使用</span><span lang="EN-US">using</span><span style="FONT-FAMILY: 宋体">子句确保实现</span><span lang="EN-US">IDisposable</span><span style="FONT-FAMILY: 宋体">接口</span><span lang="EN-US">(</span><span style="FONT-FAMILY: 宋体">详见第</span><span lang="EN-US">4</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"><span lang="EN-US">string source = &quot;server=(local)\\NetSDK;&quot; +</span></p>
<p class="a6" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; &quot; </span><span lang="EN-GB">integrated security=SSPI</span><span lang="EN-US">;&quot; + </span></p>
<p class="a6" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; &quot;database=Northwind&quot;;</span></p>
<p class="a6" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13pt; FTEL: 18.45pt"><span lang="EN-US">using ( SqlConnection conn = new SqlConnection ( source ) )</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13pt; FTEL: 18.45pt"><span lang="EN-US">{</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;&nbsp; // Open the connection</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;&nbsp; conn.Open ( ) ;</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;&nbsp; // Do something useful</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">using</span><span style="FONT-FAMILY: 宋体">子句都会确保关闭数据库连接。</span></p>
<p class="MsoNormal"><span style="FONT-FAMILY: 宋体">查看一下连接类的</span><span lang="EN-US">Dispose()</span><span style="FONT-FAMILY: 宋体">方法的</span><span lang="EN-US">IL</span><span style="FONT-FAMILY: 宋体">代码,它们都检查连接对象的当前状态,如果其状态为打开,就调用</span><span lang="EN-US">Close()</span><span style="FONT-FAMILY: 宋体">方法。浏览</span><span lang="EN-US">.NET</span><span style="FONT-FAMILY: 宋体">程序集的一个强大工具是</span><span lang="EN-US">Reflector(</span><span style="FONT-FAMILY: 宋体">可以从</span><span lang="EN-US">/www.aisto.com/roeder/dotnet/</span><span style="FONT-FAMILY: 宋体">上获得</span><span lang="EN-US">)</span><span style="FONT-FAMILY: 宋体">。这个工具允许查看任何</span><span lang="EN-US">.NET</span><span style="FONT-FAMILY: 宋体">方法的</span><span lang="EN-US">IL</span><span style="FONT-FAMILY: 宋体">代码,还可以把</span><span lang="EN-US">IL</span><span style="FONT-FAMILY: 宋体">代码反编译为源代码,让我们轻松地确定给定的方法的作用。</span></p>
<p class="MsoNormal"><span style="FONT-FAMILY: 宋体">在编程时,应至少使用这两个方法中的一个,或者两个方法都使用。无论在哪里获得资源,最好都使用</span><span lang="EN-US">using ()</span><span style="FONT-FAMILY: 宋体">语句,因为尽管我们都会编写</span><span lang="EN-US">Close()</span><span style="FONT-FAMILY: 宋体">语句,但有时会忘记,此时</span><span lang="EN-US">using</span><span style="FONT-FAMILY: 宋体">子句就会发挥作用。这两种方式都没有好的异常处理方式来替代,所以在大多数情况下,最好组合使用这两种方法,如下面的示例所示。</span></p>
<p class="2" style="MARGIN-TOP: 4.9pt; MARGIN-LEFT: 21.45pt; MARGIN-RIGHT: 0cm; FTEL: 18.45pt"><span lang="EN-US">try</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">{</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;&nbsp; using (SqlConnection conn = new SqlConnection ( source ))</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;&nbsp; {</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; // Open the connection</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; conn.Open ( ) ;</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; // Do something useful</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; // Close it myself</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; conn.Close ( ) ;</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;&nbsp; }</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">}</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">catch (Exception e)</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">{</span></p>
<p class="2" style="MARGIN-LEFT: 21.45pt; LINE-HEIGHT: 13.5pt; FTEL: 18.45pt"><span lang="EN-US">&nbsp;&nbsp; // Do something with the exception here...</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">Close()</span><span style="FONT-FAMILY: 宋体">,但这是不必要的,因为</span><span lang="EN-US">using</span><span style="FONT-FAMILY: 宋体">子句将确保在任何情况下都执行关闭操作。但是,应确保像这样的资源尽可能早地释放。因为在块的其余部分可能有更多的代码,而在这些地方没有必要锁定资源。</span></p>
<p class="MsoNormal" style="LINE-HEIGHT: 15pt"><span style="FONT-FAMILY: 宋体">另外,如果在</span><span lang="EN-US">using</span><span style="FONT-FAMILY: 宋体">块中出现了异常,</span><span lang="EN-US">using</span><span style="FONT-FAMILY: 宋体">子句就会确保在资源上调用</span><span lang="EN-US">IDisposable.Dispose</span><span style="FONT-FAMILY: 宋体">方法,在本例中将确保总是关闭数据库连接。这样,与必须确保在异常子句中关闭连接相比,代码的可读性更高。</span></p>
<p class="MsoNormal" style="LINE-HEIGHT: 15pt"><span style="FONT-FAMILY: 宋体">最后,如果编写一个封装资源的类,无论该资源是什么,都应实现</span><span lang="EN-US">IDisposable</span><span style="FONT-FAMILY: 宋体">接口,关闭资源。这样,任何使用该类的代码都可以利用</span><span lang="EN-US">using()</span><span style="FONT-FAMILY: 宋体">语句,以确保资源被释放。</span></p></div>
                <!-- page -->
                <div class="page" style="text-align: center">
                    <a href="100165596.htm">上一页</a>&nbsp;&nbsp;&nbsp;<a href="index.html">首页</a>&nbsp;&nbsp;&nbsp;<a href="100165598.htm">下一页</a>
                </div>
                <div style="margin: 0px auto; width: 700px; border: solid 1px #0b5f98;">
                    <div style="float: left; width: 16px; background-color: #0b5f98; color: White; padding: 1px;">
                        图书导读
                    </div>
                    <div style="float: right; width: 670px; text-align: left; line-height: 16pt; padding-left: 2px">
                        <!--导读-->
                        <h1 id="divCurrentNode2" style="color: #b83507; width: 100%; text-align: left; font-size: 12px; padding-left: 2px">当前章节:<a href='100165597.htm'><font color='red'>21.2.1  高效地使用连接</font></a></h1>
                        <div id="divRealteNod2" style="padding-left: 2px">
                        <div style='float:left;width:49%'>·<a href='100165594.htm'>21.1.2  共享类</a></div><div style='float:right;width:49%'>·<a href='100165595.htm'>21.1.3  数据库特定的类</a></div><div style='float:left;width:49%'>·<a href='100165596.htm'>21.2  使用数据库连接</a></div><div style='float:right;width:49%'>·<a href='100165598.htm'>21.2.2  事务处理</a></div><div style='float:left;width:49%'>·<a href='100165599.htm'>21.3  命令</a></div><div style='float:right;width:49%'>·<a href='100165600.htm'>21.3.1  执行命令</a></div></div>
                    </div>
                </div>
                </div>
            <!-- 评论 -->
            <!-- 今日推荐 -->
            </div>
        <!-- 页脚 -->
        <div id="foot">
         <img src="../../book/readbook.aspx@node=5597&bookid=16&bookname=21.2.1++_25b8_25df_25d0_25a7_25b5_25d8_25ca_25b9_25d3_25c3_25c1_25ac_25bd_25d3" alt="" width="0" height="0" />
        <style>

        </div>
</body>
</html>

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?