《C++面向对象高效编程(第2版)》——4.6 对象赋值的语义

本节书摘来自异步社区出版社《C++面向对象高效编程(第2版)》一书中的第4章,第4.6节,作者: 【美】Kayshav Dattatri,更多章节内容可以访问云栖社区“异步社区”公众号查看。

4.6 对象赋值的语义

C++面向对象高效编程(第2版)
赋值与复制的操作非常类似。在C++中,绝大多数的复制操作都由语言隐式调用(当对象按值传递或按值返回时)。当通过现有对象创建新对象时,也进行了复制操作(但不是很频繁)。与复制相反的是,赋值是必须由程序员显式调用的操作。然而,在Eiffel和Smalltalk中,赋值和复制操作都由程序员显式调用。这也是基于值的语言与基于引用的语言之间的区别。

在C++中,对于对象和基本类型赋值都具有相同的含义。把基本类型变量赋值给另一个(兼容的)基本类型变量时,将复制变量中的值。例如:

int x = 10;
int y;
y = x;```
将x中的值复制给y。同样,对于对象:

TPoint2D p1;
TPoint2D p2(100, 200);
p1 = p2;`
将p2数据成员中的值复制给p1数据成员。这里不会特别对待指针和引用,复制它们的方式和复制基本数据类型相同。这就是默认赋值操作(default assignment operation)。赋值相应成员的方法称为逐个成员赋值(memberwise assignment),它由赋值操作符实现。
`
TPoint2D::operator=(const TPoint2D& source);`
注意,operator在C++中是保留字(reserved word)1。如果类并未声明和实现该赋值操作符,编译器将自动生成一个。而且该生成的赋值操作符(称为默认赋值操作符)执行逐个成员赋值。由编译器提供的默认赋值操作符实现,类似这样:

TPoint2D::operator=(const TPoint2D& source)
{
   this->_xcoordinate = source._xcoordinate; // 复制x区域
   this->_ycoordinate = source._ycoordinate; // 复制y区域
   return *this;
}```
这看起来和前面介绍的浅复制操作非常相似。我们并不希望在TPerson类中使用这样的默认赋值操作符,因为该类包含指针,这样复制指针不安全。因此,我们将为TPerson类实现自己的赋值操作符,如下所示:

TPerson& TPerson::operator=(const TPerson& source)
{
   if (this == &source) // 自我赋值检查
    return *this;
    // 首先,复制(赋值)所有基本数据;
   this->_ssn = source._ssn;
    // 接下来,需要复制name中的字符。
    // 如果name中的空间充足,则只需复制字符即可,
    // 否则,删除name所指向的现有内存,然后分配新的内存块。
    // 最后,复制字符。
   if (source._name != 0) {  // 是否有任何复制?
    int nameLength = strlen(source._name);
    int thisNameLength = (this->_name) ?
        strlen(this->_name) : 0;    
    if (nameLength <= thisNameLength)  // 简单的情况
     strcpy(this->_name, source._name);
    else {   // 复杂的情况
     delete [] this->_name;
     name = new char[nameLength + 1];
      // +1,为放置0
     strcpy(this->_name, source._name);
    }
   }
   else {
    delete [] this->_name; this->_name = 0;
   }
    // 为address重复以上步骤
   if (source._address != 0) {
    int addressLength = strlen(source._address);
    int thisAddrLength = (this->_address) ?
        strlen(this->_address) : 0;
    if (addressLength <= thisAddrLength) {
      // 简单的情况
     strcpy(this->_address, source._address);
    }
    else {   // 复杂的情况
     delete [] this->_address;
     _address = new char[addressLength + 1];
      // +1,为放置0。
     strcpy ( this->_address, source._address);
    }
   }
   else {
    delete [] this->_address; this->_address = 0;
   }
   return *this;
}`
我们刚才实现的赋值操作符,就是将TPerson类对象显式赋值给另一个TPerson类对象时,所使用的赋值操作符。但是,某些情况下需要将其他类型的数据赋值给TPerson类对象,可以通过在类中实现重载赋值操作符,以接受不同类型的参数。在后续章节中将介绍相关的示例。还需注意的是,_birthDate数据成员不能被复制,因为它是const成员,不允许为其赋值。这里再次假设,一旦创建一个TPerson类对象,这个人的出生日期在其生存期内便不能改变。这个限制作用于_birthDate上似乎有些严格,但是它用于阐明带有const数据成员限制的复制和赋值的目的。如果这个限制对于一些应用程序而言过于严格死板,也可将_birthDate改为非const成员。

需要遵循的规则是:

hand 一定要为每个类实现赋值操作符,不要依赖语言所生成的默认赋值操作符。

感兴趣的读者可能注意到,生成的默认复制构造函数和赋值操作符都是内联函数(inline function)。欲了解C++中复制构造函数的内部细节,请参阅第13章内容。

注意:
鉴于这种情况,在C++中,很容易控制对象的赋值和复制。如果我们希望禁止公有客户和派生类复制对象,只需将复制构造函数设置成private。对于赋值操作符也一样。还需注意,可以限制(而不是完全禁止)制作的副本数目。本章稍后将会解释为什么需要这种副本控制。
Smalltalk:

了解Smalltalk中的赋值语义很有趣。该语言用<-操作符表示赋值。例如,a <- b`意味着将b赋值给a。对于简单(或者基本)类型,赋值所涉及的复制值和C++中一样。但是,赋值应用于对象时,行为则完全不同。在Smalltalk中,每个对象都是对内存中其他对象的引用。鉴于此,对象赋值实际上就是引用赋值。如果a和b都是对象,且通过操作a <- b将b赋值给a,则b所引用的对象通过名称a获得另一个引用。赋值后,a和b都引用相同的对象。无需多说,无论a在赋值之前所引用的是什么,在赋值后都不能通过a再访问它。注意,Smalltalk不允许在函数内部对形参赋值,这是该语言的一项限制。

Eiffel:

Eiffel在赋值限制方面和Smalltalk完全一样。该语言用操作符:=表示赋值(和Pascal一样)。同样,简单类型间的赋值也指复制值,而对象(实际上是对象引用( object reference ))间的赋值则意味着复制引用。和Smalltalk一样,Eiffel也不允许对形参赋值。

4.6.1 左值操作赋值
左值就是可被修改的值(通常在赋值操作符左侧的名称)。详见第3章中对左值和右值的介绍。在C和C++中,默认赋值语义会产生一个左值,这意味着可以进行级联赋值操作(即,a = b = c)。在Smlltalk中也可以这样。实际上,Smalltalk中的每种方法都保证有返回值,这是赋值的有用副作用,利用它可以写出清楚简练的表达式。但是,Eiffel却不允许级联赋值,这和Pascal一样。

记住:

C++允许实现者定义复制对象的语义。
C++允许实现者定义赋值的语义。
实现者可以在每个类的基础上控制复制和赋值语义。
1译者注:现在,operator在C++中已成为关键字。
本文仅用于学习和交流目的,不代表异步社区观点。非商业转载请注明作译者、出处,并保留本文的原始链接。

### 回答1: 《C++面向对象高效编程 第2》是一本关于C++编程的书籍。本书主要介绍了面向对象编程的概念、原则和技巧,以及如何利用C++语言进行高效面向对象编程。 在书中,作者详细讲解了C++面向对象特性,包括封装、继承和多态等。同时,还介绍了C++中的其他重要特性,如模板、异常处理和STL库等。这些知识点的讲解结合了大量的例子和实践,使读者能够更好地理解和应用这些概念。 本书在第2中进行了更新和扩充,添加了新的内容,涉及了现代C++的一些特性和技术,如智能指针、移动语义和并发编程等。这些新的内容可以帮助读者更好地应对现代软件开发的需求和挑战。 与此同时,本书还提供了大量的编程实践和案例,帮助读者掌握实际的编程技巧和经验。通过这些实践,读者可以学习到如何编写高效、可复用和易于维护的C++代码。 总之,《C++面向对象高效编程 第2》是一本适合C++初学者和有一定经验的程序员阅读的书籍。它不仅介绍了面向对象编程的基本概念和技巧,还提供了大量的实例和编程实践,帮助读者掌握C++语言的高效应用。无论是想深入学习C++面向对象编程的基础知识,还是想了解C++中一些高级特性的使用,这本书都是一本值得推荐的参考书。 ### 回答2: 《C++面向对象高效编程 第2》是一本关于C++面向对象编程的经典教材。本书的PDF本可以在网络上进行搜索和获取。该书全面介绍了对象、类、继承、多态等C++面向对象编程的基础知识,同时还深入讲解了C++高效编程技巧和最佳实践。 在本书中,作者详细解析了C++的语法和特性,并提供了大量的示例代码和演示,以帮助读者更好地理解和掌握面向对象编程的概念和技术。此外,本书还介绍了一些高级主题,如异常处理、模板编程、STL等,在帮助读者提高编程能力的同时,也为读者拓宽了编程思维。 通过学习《C++面向对象高效编程 第2PDF,读者可以系统地学习C++面向对象编程的知识,深入理解面向对象编程的原理和思想,并能够灵活运用这些知识进行开发和设计。同时,本书还提供了一些经验和技巧,帮助读者编写出高效、可维护的C++代码。 总而言之,该书是一本重要的学习资料,不仅适合初学者入门,也适合有一定C++编程基础的开发者进一步提升自己的编程水平。通过阅读《C++面向对象高效编程 第2PDF,读者将能够系统学习C++面向对象编程,并在实际开发中运用所学知识,提升自己的编程能力。 ### 回答3: 《面向对象高效编程 第2》是一本关于面向对象编程的书籍,主要介绍了面向对象编程的原理、思想和最佳实践。 在这本书中,作者首先介绍了面向对象编程的基本概念,包括类、对象、继承、多态等。然后,作者详细介绍了如何有效地使用面向对象编程进行软件开发。他提出了一些实用的技巧和方法,帮助读者更好地理解和应用面向对象编程的思想。 除了介绍基本的面向对象编程理论外,这本书还提供了大量的案例和示例代码,帮助读者实际运用所学知识解决实际问题。这些案例涵盖了不同领域的应用,包括图形界面、网络编程、数据库操作等,让读者能够从不同角度了解面向对象编程的实际应用场景。 此外,这本书还介绍了一些常用的面向对象编程语言,如Java、C++和Python等。作者通过比较不同的编程语言特点和应用场景,帮助读者选择合适的编程语言,并指导读者如何在具体的编程语言中高效地实践面向对象编程。 总之,《面向对象高效编程 第2》是一本很好的面向对象编程学习资料。无论是初学者还是有一定经验的开发者,都能从中获取到宝贵的学习经验和实践技巧。通过阅读这本书,读者能够对面向对象编程有更深入的理解,提升自己在软件开发中的能力。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值