📄 effective c++ 2e item17.htm
字号:
函数定义参见条款11<BR>
// <BR> ...</P>
<P> String& operator=(const String& rhs);</P>
<P>private:<BR> char *data;<BR>};</P>
<P>// 忽略了给自己赋值的情况<BR>// 的赋值运算符<BR>String& String::operator=(const
String& rhs)<BR>{<BR> delete [] data; // delete old
memory</P>
<P> // 分配新内存,将rhs的值拷贝给它<BR> data = new char[strlen(rhs.data) +
1];<BR> strcpy(data, rhs.data);</P>
<P> return *this; // see Item 15<BR>}</P>
<P>看看下面这种情况将会发生什么:</P>
<P>String a = "Hello";</P>
<P>a =
a;
// same as a.operator=(a)</P>
<P>赋值运算符内部,*this和rhs好象是不同的对象,但在现在这种情况下它们却恰巧</P>
<P>是同一个对象的不同名字。可以这样来表示这种情况:</P>
<P>*this data ------------>
"Hello\0"<BR>
/<BR>
/<BR>rhs data -----</P>
<P>赋值运算符做的第一件事是用delete删除data,其结果将如下所示:</P>
<P>*this data ------------>
???<BR>
/<BR>
/<BR>rhs data -----</P>
<P>现在,当赋值运算符对rhs.data调用strlen时,结果将无法确定。这是因为data被</P>
<P>删除的时候rhs.data也被删除了,data,this->data 和rhs.data 其实都是同一个</P>
<P>指针!从这一点看,情况只会越变越糟糕。</P>
<P>现在可以知道,解决问题的方案是对可能发生的自己给自己赋值的情况先进行检查</P>
<P>,如果有这种情况就立即返回。不幸的是,这种检查说起来容易做起来难,因为你</P>
<P>必须定义两个对象怎么样才算是“相同”的。</P>
<P>你面临的这个问题学术上称为object identity,它在面向对象领域是个很有名的</P>
<P>论题。本书不是讲述object identity的地方,但有必要提到两个解决这个问题的</P>
<P>基本方法。</P>
<P>一个方法是,如果两个对象具有相同的值,就说它们是相同的(具有相同的身份)</P>
<P>。例如,两个String对象如果都表示的是相同顺序的字符序列,它们就是相同的:</P>
<P>String a = "Hello";<BR>String b = "World";<BR>String c = "Hello";</P>
<P>a和c具有相同值,所以它们被认为是完全相同的;b和它们都不同。如果把这个定</P>
<P>义用到String类中,赋值运算符看起来就象这样:</P>
<P>String& String::operator=(const String& rhs)<BR>{<BR> if
(strcmp(data, rhs.data) == 0) return *this;</P>
<P> ...</P>
<P>}</P>
<P>值相等通常由operator==来检测,所以对于一个用值相等来检测对象身份的类C来</P>
<P>说,它的赋值运算符的一般形式是:</P>
<P>C& C::operator=(const C& rhs)<BR>{<BR> // 检查对自己赋值的情况<BR>
if (*this ==
rhs) //
假设operator=存在<BR> return *this;</P>
<P> ...</P>
<P>}</P>
<P>注意这个函数比较的是对象(通过operator=),而不是指针。用值相等来确定对</P>
<P>象身份和两个对象是否占用相同的内存没有关系;有关系的只是它们所表示的值。</P>
<P>另一个确定对象身份是否相同的方法是用内存地址。采用这个定义,两个对象当且</P>
<P>仅当它们具有相同的地址时才是相同的。这个定义在C++程序中运用更广泛,可能</P>
<P>是因为它很容易实现而且计算很快,而采用值相等的定义则不一定总具有这两个优</P>
<P>点。采用地址相等的定义,一个普通的赋值运算符看起来象这样:</P>
<P>C& C::operator=(const C& rhs)<BR>{<BR> // 检查对自己赋值的情况<BR>
if (this == &rhs) return *this;</P>
<P> ...</P>
<P>}</P>
<P>它对很多程序都适用。</P>
<P>如果需要一个更复杂的机制来确定两个对象是否相同,这就要靠程序员自己来实现</P>
<P>。最普通的方法是实现一个返回某种对象标识符的成员函数:</P>
<P>class C {<BR>public:<BR> ObjectID identity() const;
// 参见条款36</P>
<P> ...</P>
<P>};</P>
<P>对于两个对象指针a和b,当且仅当 a->identity() == b->identity()的时候,它</P>
<P>们所指的对象是完全相同的。当然,必须自己来实现ObjectIDs的operator==。</P>
<P>别名和object identity的问题不仅仅局限在operator=里。在任何一个用到的函数</P>
<P>里都可能会遇到。在用到引用和指针的场合,任何两个兼容类型的对象名称都可能</P>
<P>指的是同一个对象。下面列出的是别名出现的其它情形:</P>
<P>class Base {<BR> void mf1(Base&
rb); //
rb和*this可能相同
<BR> ...</P>
<P>};</P>
<P>void f1(Base& rb1,Base& rb2); //
rb1和rb2可能相同<BR>
// </P>
<P>class Derived: public Base {<BR> void mf2(Base&
rb); //
rb和*this可能相同<BR>
// <BR> ...</P>
<P>};</P>
<P>int f2(Derived& rd, Base& rb); //
rd和rb可能相同<BR>
// </P>
<P>这些例子刚好都用的是引用,指针也一样。</P>
<P>可以看到,别名可以以各种形式出现,所以决不要忘记它或期望自己永远不会碰到</P>
<P>它。也许你不会碰到,但我们大多数会碰到。而很明显的一条是,处理它会达到事</P>
<P>半功倍的效果。所以任何时候写一个函数,只要别名有可能出现,就必须在写代码</P>
<P>时进行处理。</P><BR><BR></DIV></DIV></DIV><BR><BR>
<SCRIPT src="Effective C++ 2e Item17.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=8744> 评论人: <INPUT name=csdnname>
密码: <INPUT name=csdnpassword type=password>
评论:<BR> <TEXTAREA cols=100 name=critique_content rows=8></TEXTAREA><BR>
<INPUT name=ubmit onclick=javascript:submit1(); type=button value=发表评论>
<INPUT name=id type=hidden value=8744> </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 + -