【Effective C++】item 21: Don‘t try to return a reference when you must return an object.

Don’t try to return a reference when you must return an object.

必须返回对象时,别妄想返回其reference。

考虑如下代码

class Rational
{
private:
    /* data */
    int n, d; // 分子,分母
    friend const Rational operator* (const Rational& lhs, const Rational& rhs);
public:
    Rational(int numerator = 0, int denominator = 1);
    ~Rational();
};

Rational::Rational(int numerator, int denominator): n(numerator), d(denominator) {}

Rational::~Rational() {}

如何实现其乘法的运算符重载?

  1. 在stack空间创建对象,返回其引用
const Rational& operator* (const Rational& lhs, const Rational& rhs) // stack上创建对象,返回其reference
{
    Rational result(lhs.n * rhs.n, lhs.d * rhs.d);
    return result;
}

reference指向了某个local对象,而local对象在函数退出之前被销毁了。(返回指针,指向某个local对象也是一样)

  1. 在heap空间创建对象,返回其引用
const Rational& operator* (const Rational& lhs, const Rational& rhs) // heap上创建对象,返回其reference
{
    Rational* result = new Rational(lhs.n * rhs.n, lhs.d * rhs.d);
    return *result;
}

谁该对着你new出来的对象实施delete

  1. 使用static对象,返回其引用
const Rational& operator* (const Rational& lhs, const Rational& rhs) // static对象,返回其reference
{
    static Rational result;
    result.assign(lhs.n * rhs.n, lhs.d * rhs.d);
    return result;
}

问题1. 线程安全隐患

问题2. 考虑到如下代码

bool operator== (const Rational& lhs, const Rational& rhs)
{
    return lhs.n * rhs.d == lhs.d * rhs.n;
}
bool flag = ((a * b) == (c * d));
cout << flag << endl; // flag恒为true

static对象永远是”最新“值。

总结

  1. 绝不要返回pointer或者reference指向一个local stack对象
  2. 或者reference指向一个heap-allocated对象
  3. 或者返回pointer或reference指向一个local static对象而可能需要多个这样的对象
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值