Effective c++学习笔记——条款10:令operator=返回一个*this的引用

本文深入探讨了C++中赋值操作符返回*this的协议及其背后的原因,通过实例展示了如何实现连锁赋值,并强调了遵循此协议对于类设计的必要性。
Have assignment operators return a reference to *this
从题目,我们就要记住这条信息,让你的operator=函数return *this;
基本类型int、char等都提供了连锁赋值,并采用右结合律。
int x, y, z;
x = y = z = 15; 这句真正被解析为:x = (y = (z = 15));
为了实现连锁赋值,赋值操作符必须返回一个引用指向操作符左侧实参,这是通常在C++中为classes所遵循的协议。
下面看一个实际例子;
// have_assi.cpp : 定义控制台应用程序的入口点。 //2011/9/10 by wallwind on sunrise #include "stdafx.h" #include <iostream> using namespace std; class Widget { public: Widget():i(0){} Widget(int ii):i(ii){} Widget& operator=(const Widget &rhs) { this->setValue(rhs.getValue()); return *this; } Widget& operator+=(const Widget& rhs) { this->setValue(rhs.getValue() + this->getValue()); return *this; } int getValue() const {return i;} void setValue(int ii){i=ii;} private: int i; }; int _tmain(int argc, _TCHAR* argv[]) { Widget w1,w2,w3; w1=w2=w3=10; cout<<w1.getValue()<<endl; cout<<w2.getValue()<<endl; cout<<w3.getValue()<<endl; w1+=10; w2+=20; w3+=30; cout<<w1.getValue()<<endl; cout<<w2.getValue()<<endl; cout<<w3.getValue()<<endl; return 0; }
输出结果,如图所示
如书中所说,这只是个协议,并我强制性,如果不遵守,代码一样可以通过他编译。然而c++的内置类型,和stl如
string vector 等都遵守这个协议。所以,你最好按这个做吧。

请记住:

令赋值(assignment)操作符返回一个reference to *this。

你给出的代码片段是 `CMtfRslt` 类中的两个赋值操作符重载函数的框架。为了补全这些函数,我们需要了解 `CMtfRslt` 类的结构和目的,但由于没有提供类的定义,我将基于常见 C++ 编程实践来补全这两个函数,并解释它们的用途。 --- ### 1. `CMtfRslt & CMtfRslt::operator=(const CMtfRslt & src)` 这是一个**拷贝赋值操作符**的声明和定义。它用于将一个对象的值复制到另一个已经存在的对象中。 ### 2. `CMtfRslt * CMtfRslt::operator=(const CMtfRslt * src)` 这是一个**指针赋值操作符重载**。它不太常见,可能是为了某种特殊用途(例如资源管理、智能指针模拟等)而设计的。通常返回 `this` 指针是合理的,但具体实现取决于设计意图。 --- ### 补全代码如下: ```cpp // 拷贝赋值操作符 CMtfRslt & CMtfRslt::operator=(const CMtfRslt & src) { if (this != &src) // 防止自赋值 { // 假设 m_data 是一个内部资源,比如动态分配的内存或某种结构体 // 示例:深拷贝一个数据结构 delete[] m_data; // 如果 m_data 是 new[] 分配的 m_data = new char[strlen(src.m_data) + 1]; strcpy(m_data, src.m_data); // 其他成员变量拷贝 m_size = src.m_size; m_valid = src.m_valid; } return *this; } // 指针赋值操作符 CMtfRslt * CMtfRslt::operator=(const CMtfRslt * src) { if (this != src) // 防止自赋值 { // 假设我们希望*this的内容被src指向的对象内容覆盖 *this = *src; // 调用拷贝赋值操作符 } return this; } ``` --- ### 解释: 1. **拷贝赋值操作符 (`operator=(const CMtfRslt&)`)**: - 用于对象之间的赋值。 - 需要处理自赋值(`this != &src`)。 - 如果类中包含动态资源(如指针),必须进行深拷贝,否则会出现浅拷贝问题(如重复释放内存)。 2. **指针赋值操作符 (`operator=(const CMtfRslt*)`)**: - 这个操作符的设计不太常规。 - 它接收一个指针,并返回当前对象的地址。 - 上面的实现中,我们将 `*this` 的内容用 `*src` 赋值,相当于调用拷贝赋值操作符。 - 返回 `this` 是为了支持链式赋值。 --- ### 注意事项: - 如果 `CMtfRslt` 是一个资源管理类(如智能指针、容器等),则需要特别注意资源释放和拷贝语义。 - 如果类中包含指针或资源句柄,务必进行深拷贝或使用智能指针(如 `std::unique_ptr`, `std::shared_ptr`)。 - 若 `operator=(const CMtfRslt*)` 是为了实现某种特殊的赋值逻辑(如资源转移),应根据具体需求调整。 --- ###
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值