⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 effective c++ 2e item31.htm

📁 Effective-c++.国外很经典的一本关于c++编程的电子书。
💻 HTM
📖 第 1 页 / 共 2 页
字号:
<DIV class=fstdiv3 id=print2><BR><BR>
<P>条款31: 千万不要返回局部对象的引用,也不要返回函数内部用new初始化的指针的引用</P>
<P>本条款听起来很复杂,其实不然。它只是一个很简单的道理,真的,相信我。</P>
<P>先看第一种情况:返回一个局部对象的引用。它的问题在于,局部对象 ----- 顾名思义 ---- 
仅仅是局部的。也就是说,局部对象是在被定义时创建,在离开生命空间时被销毁的。所谓生命空间,是指它们所在的函数体。当函数返回时,程序的控制离开了这个空间,所以函数内部所有的局部对象被自动销毁。因此,如果返回局部对象的引用,那个局部对象其实已经在函数调用者使用它之前被销毁了。</P>
<P>当想提高程序的效率而使函数的结果通过引用而不是值返回时,这个问题就会出现。下面的例子和条款23中的一样,其目的在于详细说明什么时候该返回引用,什么时候不该:</P>
<P>class Rational {&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; // 
一个有理数类<BR>public:<BR>&nbsp; Rational(int numerator = 0, int denominator = 
1);<BR>&nbsp; ~Rational();</P>
<P>&nbsp; ...</P>
<P>private:<BR>&nbsp; int n, 
d;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
// 分子和分母</P>
<P>// 注意operator* (不正确地)返回了一个引用<BR>friend const Rational&amp; operator*(const 
Rational&amp; 
lhs,<BR>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
const Rational&amp; rhs);<BR>};</P>
<P>// operator*不正确的实现<BR>inline const Rational&amp; operator*(const 
Rational&amp; 
lhs,<BR>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
const Rational&amp; rhs)<BR>{<BR>&nbsp; Rational result(lhs.n * rhs.n, lhs.d * 
rhs.d);<BR>&nbsp; return result;<BR>}</P>
<P>这里,局部对象result在刚进入operator*函数体时就被创建。但是,所有的局部对象在离开它们所在的空间时都要被自动销毁。具体到这个例子来说,result是在执行return语句后离开它所在的空间的。所以,如果这样写:</P>
<P>Rational two = 2;</P>
<P>Rational four = two * two;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; // 
同operator*(two, two)</P>
<P><BR>函数调用时将发生如下事件:</P>
<P>1. 局部对象result被创建。<BR>2. 
初始化一个引用,使之成为result的另一个名字;这个引用先放在另一边,留做operator*的返回值。<BR>3. 
局部对象result被销毁,它在堆栈所占的空间可被本程序其它部分或其他程序使用。<BR>4. 用步骤2中的引用初始化对象four。</P>
<P>一切都很正常,直到第4步才产生了错误,借用高科技界的话来说,产生了"一个巨大的错误"。因为,第2步被初始化的引用在第3步结束时指向的不再是一个有效的对象,所以对象four的初始化结果完全是不可确定的。</P>
<P>教训很明显:别返回一个局部对象的引用。</P>
<P>"那好,"你可能会说,"问题不就在于要使用的对象离开它所在的空间太早吗?我能解决。不要使用局部对象,可以用new来解决这个问题。"象下面这样:</P>
<P>// operator*的另一个不正确的实现<BR>inline const Rational&amp; operator*(const 
Rational&amp; 
lhs,<BR>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
const Rational&amp; rhs)<BR>{<BR>&nbsp; // create a new object on the 
heap<BR>&nbsp; Rational *result =<BR>&nbsp;&nbsp;&nbsp; new Rational(lhs.n * 
rhs.n, lhs.d * rhs.d);</P>
<P>&nbsp; // return it<BR>&nbsp; return *result;<BR>}</P>
<P>这个方法的确避免了上面例子中的问题,但却引发了新的难题。大家都知道,为了在程序中避免内存泄漏,就必须确保对每个用new产生的指针调用delete,但是,这里的问题是,对于这个函数中使用的new,谁来进行对应的delete调用呢?</P>
<P>显然,operator*的调用者应该负责调用delete。真的显然吗?遗憾的是,即使你白纸黑字将它写成规定,也无法解决问题。之所以做出这么悲观的判断,是基于两条理由:</P>
<P>第一,大家都知道,程序员这类人是很马虎的。这不是指你马虎或我马虎,而是指,没有哪个程序员不和某个有这类习性的人打交道。想让这样的程序员记住无论何时调用operator*后必须得到结果的指针然后调用delete,这样的几率有多大呢?也是说,他们必须这样使用operator*:</P>
<P>const Rational&amp; four = two * two;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; // 
得到废弃的指针;<BR>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
// 将它存在一个引用中<BR>...</P>
<P>delete 
&amp;four;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
// 得到指针并删除</P>
<P>这样的几率将会小得不能再小。记住,只要有哪怕一个operator*的调用者忘了这条规则,就会造成内存泄漏。</P>
<P>返回废弃的指针还有另外一个更严重的问题,即使是最尽责的程序员也难以避免。因为常常有这种情况,operator*的结果只是临时用于中间值,它的存在只是为了计算一个更大的表达式。例如:</P>
<P>Rational one(1), two(2), three(3), four(4);<BR>Rational product;</P>
<P>product = one * two * three * four;</P>
<P>product的计算表达式需要三个单独的operator*调用,以相应的函数形式重写这个表达式会看得更清楚:</P>
<P>product = operator*(operator*(operator*(one, two), three), four);</P>
<P>是的,每个operator*调用所返回的对象都要被删除,但在这里无法调用delete,因为没有哪个返回对象被保存下来。</P>
<P>解决这一难题的唯一方案是叫用户这样写代码:</P>
<P>const Rational&amp; temp1 = one * two;<BR>const Rational&amp; temp2 = temp1 * 
three;<BR>const Rational&amp; temp3 = temp2 * four;</P>
<P>delete &amp;temp1;<BR>delete &amp;temp2;<BR>delete &amp;temp3;</P>
<P>果真如此的话,你所能期待的最好结果是人们将不再理睬你。更现实一点,你将会在指责声中度日,或者可能会被判处10年苦力去写威化饼干机或烤面包机的微代码。</P>
<P>所以要记住你的教训:写一个返回废弃指针的函数无异于坐等内存泄漏的来临。</P>
<P>另外,假如你认为自己想出了什么办法可以避免"返回局部对象的引用"所带来的不确定行为,以及"返回堆(heap)上分配的对象的引用"所带来的内存泄漏,那么,请转到条款23,看看为什么返回局部静态(static)对象的引用也会工作不正常。看了之后,也许会帮助你避免头痛医脚所带来的麻烦。</P><BR><BR></DIV></DIV></DIV><BR><BR>
<SCRIPT src="Effective C++ 2e Item31.files/get_readnum.htm"></SCRIPT>

<TABLE align=center bgColor=#666666 border=0 cellPadding=2 cellSpacing=1 
width=770>
  <TBODY>
  <TR>
    <TH bgColor=#990000 id=white><FONT 
  color=#ffffff>对该文的评论</FONT></TH></TR></TBODY></TABLE><BR>
<SCRIPT language=javascript>
<!--
function isEmpty(s)
{  
	return ((s == null) || (s.length == 0))
}
function submit1()
{
   if (isEmpty(document.add_critique.csdnname.value) || isEmpty(document.add_critique.csdnpassword.value) || isEmpty(document.add_critique.critique_content.value))
   {
      alert('登陆名,密码,评论不能为空!!!!')   ;
      return false;
   }
   document.add_critique.submit();
 }
//-->
</SCRIPT>

<TABLE align=center bgColor=#666666 border=0 cellPadding=2 cellSpacing=1 
width=770>
  <TBODY>
  <TR>
    <TH bgColor=#990000 id=white><FONT 
color=#ffffff>发表评论</FONT></TH></TR></TBODY></TABLE>
<TABLE align=center bgColor=#ffffff border=0 cellPadding=2 cellSpacing=1 
width=770>
  <TBODY>
  <TR>
    <TD>
      <FORM action=/develop/add_critique.asp method=post 
      name=add_critique><INPUT name=critique_add type=hidden value=add> <INPUT 
      name=from type=hidden value=9030> &nbsp;&nbsp;评论人: <INPUT name=csdnname> 
      &nbsp;&nbsp;密码: <INPUT name=csdnpassword type=password> 
      &nbsp;&nbsp;评论:<BR>&nbsp;&nbsp; <TEXTAREA cols=100 name=critique_content rows=8></TEXTAREA><BR>&nbsp;&nbsp; 
<INPUT name=ubmit onclick=javascript:submit1(); type=button value=发表评论> 
      <INPUT name=id type=hidden value=9030> </FORM></TD></TR></TBODY></TABLE>
<TABLE border=0 width=770>
  <TBODY>
  <TR>
    <TD>
      <TABLE border=0 cellPadding=2 width=770>
        <TBODY>
        <TR>
          <TD height=10></TD></TR>
        <TR>
          <TD align=left width=130><A 
            href="http://www.csdn.net/news/looknews.asp?id=2313" 
            target=_blank><IMG border=0 
            src="http://www.csdn.net/job/images/csdn_job.gif"></A> </TD>
          <TD align=middle width=510>
            <OBJECT classid=clsid:D27CDB6E-AE6D-11cf-96B8-444553540000 
            codeBase=http://active.macromedia.com/flash2/cabs/swflash.cab#version=4,0,0,0 
            height=60 id=Movie1 width=468><PARAM NAME="movie" VALUE="http://www.csdn.net/images/ad/csdn_media.swf"><PARAM NAME="quality" VALUE="high">
               <EMBED src="http://www.csdn.net/images/ad/csdn_media.swf" 
            quality=high WIDTH=468 HEIGHT=60 
            TYPE="application/x-shockwave-flash" 
            PLUGINSPAGE="http://www.macromedia.com/shockwave/download/index.cgi?P1_Prod_Version=ShockwaveFlash"> 
            </EMBED> </OBJECT></TD>
          <TD align=middle width=130><A 
            href="http://www.hd315.gov.cn/beian/view.asp?bianhao=010202001032100010"><IMG 
            border=0 src="http://www.csdn.net/images/biaoshi.gif"></A> 
        </TD></TR></TBODY></TABLE></TD></TR>
  <TR>
    <TD>
      <TABLE align=default bgColor=#1f60a0 border=0 cellPadding=0 cellSpacing=0 
      height=20 width=770>
        <TBODY>
        <TR align=middle class=font13px vAlign=center>
          <TD class=font13px height=2 vAlign=center width=90></TD>
          <TD rowSpan=2 width=2><IMG height=13 
            src="http://www.csdn.net/images/ad4.gif" width=2></TD>
          <TD class=font13px height=2 width=90></TD>
          <TD rowSpan=2 width=3><FONT color=#ffffff><IMG height=13 
            src="http://www.csdn.net/images/ad4.gif" width=2></FONT></TD>
          <TD class=font13px height=2 width=90></TD>
          <TD rowSpan=2 width=2><IMG height=13 
            src="http://www.csdn.net/images/ad4.gif" width=2></TD>
          <TD class=font13px height=2 width=90></TD>
          <TD rowSpan=2 width=3><FONT color=#ffffff><IMG height=13 
            src="http://www.csdn.net/images/ad4.gif" width=2></FONT></TD>
          <TD height=2 width=350><FONT color=#ffffff></FONT></TD>
          <TD rowSpan=2 width=10><FONT color=#ffffff><IMG height=11 
            src="http://www.csdn.net/images/ad3.gif" width=6></FONT></TD>
          <TD class=font13px height=2 width=60></TD></TR>
        <TR align=middle class=font14px vAlign=center>
          <TD class=font13px vAlign=center width=90><A 
            href="http://www.csdn.net/intro/intro.shtm"><FONT 
            color=#ffffff>美达美简介</FONT></A></TD>
          <TD class=font13px width=90><A 
            href="http://www.csdn.net/intro/ad.shtm"><FONT 
            color=#ffffff>广告服务</FONT></A></TD>
          <TD class=font13px width=90><A 
            href="http://www.csdn.net/English/"><FONT 
            color=#ffffff>英语步步高</FONT></A></TD>
          <TD class=font13px width=90><A href="http://www.csdn.net/dev/"><FONT 
            color=#ffffff>程序员大本营</FONT></A></TD>
          <TD align=right width=350><SPAN class=font13px><A 
            href="mailto:webmaster@csdn.net"><FONT 
            color=#ffffff>百联美达美科技有限公司</FONT></A></SPAN><FONT color=#ffffff><SPAN 
            class=font13px> </SPAN></FONT></TD>
          <TD class=font13px width=60><FONT color=#ffffff>版权所有</FONT>
            <SCRIPT>document.write("<img src=http://202.106.156.10/stat.asp?user=designol&refer="+escape(document.referrer)+"&cur="+escape(document.URL)+" width=0 height=0 border=0>");</SCRIPT>
          </TD></TR></TBODY></TABLE></TD></TR></TBODY></TABLE></CENTER></BODY></HTML>

⌨️ 快捷键说明

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