const对象竟然可以修改其成员,只因顶层const

评论区的大哥的解释很有道理,在下面贴出来,我收回之前的言论:
class::func( ) const成员函数的返回值是string& 不是const string&,这是问题所在。
class::func( ) const内的一切类成员都是constthis指针是const*,ps类型是const string*return*ps类型是const string返回为string&,有些编译器会警告报错。请严格控制返回值类型是否const,非const_cast<>不解const

==================== 以上2022.11.5更新

我自定义了一个类 HasPtr,并创建了一个它的常量对象,get_str 返回成员的引用。

#include <iostream>
#include "HasPtr_ex30test.h"

using std::cout;

int main()
{
    const HasPtr hp(0, "hello");
    cout << hp.get_str() << '\n';
    hp.get_str() = "jesus";
    cout << hp.get_str() << '\n';
    return 0;
}

输出:

hello
jesus

看到这一幕真的很震惊,const 对象的成员竟然被修改了。这究竟怎么回事,下面给出头文件。

#ifndef HASPTR_EX30TEST_H
#define HASPTR_EX30TEST_H

#include <string>

using std::string;

class HasPtr
{
    friend void swap(HasPtr &, HasPtr &);

public:
    HasPtr(int i_, const string &s = "") : ps(new string(s)), i(i_) {}
    HasPtr(const HasPtr &hp) : ps(new string(*hp.ps)), i(hp.i) {}
    HasPtr &operator=(const HasPtr &hp);
    ~HasPtr() { delete ps; }
    string &get_str() const
    {
        return *ps;
    }

private:
    string *ps;
    int i;
};
// 其余函数
#endif

关键就在 get_str() 成员函数上,如果 ps 是个 string 对象,肯定是不能修改的了,但偏偏它是一个 string 指针。这会发生什么呢?

return *ps; 相当于 return *((*this).ps) 。我们知道这是一个常量成员函数,所以 *this 是常量,所以 (*this).ps 也是常量,也是一个 string 指针。问题来了,这个常量是顶层还是底层?答案是顶层。如果不信,我们做如下改动:

    string &get_str() const
    {
        *ps = "ggg";
        return *ps;
    }

main 函数不变,输出:

ggg

ggg

可以看到在常量成员函数中成功修改了 *ps 的值,是不是不可思议。其实不能修改的是 ps ,但 *ps 确实可以修改,这也印证了 ps 就是顶层 const

*((*this).ps) 是对顶层 const 解引用,当然也就丢弃了常量特性了。所以非常量可以绑定到 string & 上面,进而可以在 main 函数中修改为 "jesus"

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 4
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值