c++中如何使用mutable关键字_c++常函数修改成员变量方法

发布时间 - 2026-01-03 00:00:00    点击率:
mutable成员变量可在const成员函数中修改,用于缓存、计数、锁等辅助状态,不破坏逻辑常量性;必须声明在类内,不可与static/const共用,不影响内存布局,但需确保所有相关字段均标记为mutable。

mutable 成员变量可以在 const 成员函数中被修改

当一个成员函数声明为 const,编译器默认禁止修改任何非静态成员变量。但有些场景下,你确实需要在逻辑“只读”的函数里更新某些辅助状态(比如缓存、访问计数、互斥锁),这时 mutable 就是唯一合法出口。

它不改变对象的逻辑常量性,只告诉编译器:“这个变量允许被 const 函数修改”。关键点是:它必须用于成员变量声明处,且仅对类内变量有效。

  • mutable 不能修饰局部变量、全局变量或函数参数
  • 不能与 staticconst 同时修饰同一个变量(mutable static int x; 是非法的)
  • 即使变量是 mutable,也不能绕过线程安全——多线程下仍需同步(如用 mutable std::mutex mtx; 配合 mtx.lock()

典型使用场景:缓存计算结果

避免重复开销较大的计算,又不想破坏接口的 const 语义。例如字符串长度缓存:

class StringWrapper {
    std::string data_;
    mutable size_t len_cache_;
    mutable bool cache_valid_;

public:
    StringWrapper(const std::string& s) : data_(s), len_cache_(0), cache_valid_(false) {}

    size_t length() const {
        if (!cache_valid_) {
            len_cache_ = data_.length();  // ✅ 允许修改 mutable 成员
            cache_valid_ = true;
        }
        return len_cache_;
    }
};

注意:len_cache_cache_valid_ 都必须声明为 mutable,否则在 length() 中赋值会触发编译错误:assignment of member 'StringWrapper::cache_valid_' in a const member function

立即学习“C++免费学习笔记(深入)”;

和 const_cast 的本质区别

有人试图用 const_cast 强转 this 指针来修改普通成员变量,这是危险且通常不合法的:

  • 如果原对象本身是 const(比如传入 const StringWrapper&),通过 const_cast 修改会引发未定义行为(UB)
  • mutable 是语言明确支持的例外机制,编译器和优化器都认可其合法性
  • const_cast 绕过类型系统,而 mutable 是设计契约的一部分

所以:要用缓存、计数、锁等辅助状态,就老实用 mutable;别碰 const_cast 去改非 mutable 成员。

容易忽略的细节:mutable 不影响对象布局或性能

mutable 是纯编译期标记,不改变变量的内存位置、大小或访问速度。但它会影响代码可读性和维护性:

  • 过度使用会让“哪些状态真可变”变得模糊,建议只用于真正无副作用的辅助字段
  • 调试时注意:const 对象的 mutable 成员仍可能变化,Watch 表达式里看到它的值跳变是正常的
  • 序列化/拷贝构造时,mutable 成员仍参与,不会被自动忽略

最常被漏掉的是忘记把所有相关字段都标为 mutable —— 比如加了 mutable int count; 却忘了 mutable bool dirty;,一写就报错。


# app  # c++  # 区别  # 编译错误  # 代码可读性  # Static  # 常量  # count  # 成员变量  # 成员函数  # const  # 局部变量  # 全局变量  # 字符串  # bool  # int  # mutable  # 指针  # 接口  # Length  # 线程  # 多线程  # function  # 对象  # this  # 的是  # 这是  # 可在  # 要用  # 会让  # 又不  # 报错  # 它会  # 则在  # 它不 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel怎么调用外部API_Laravel Http Client客户端使用  小米17系列还有一款新机?主打6.9英寸大直屏和旗舰级影像  Win11关机界面怎么改_Win11自定义关机画面设置【工具】  canvas 画布在主流浏览器中的尺寸限制详细介绍  如何彻底卸载建站之星软件?  Laravel如何使用Guzzle调用外部接口_Laravel发起HTTP请求与JSON数据解析【详解】  JavaScript数据类型有哪些_如何准确判断一个变量的类型  PHP正则匹配日期和时间(时间戳转换)的实例代码  Laravel如何处理文件下载请求?(Response示例)  大连 网站制作,大连天途有线官网?  Swift中switch语句区间和元组模式匹配  Laravel如何与Pusher实现实时通信?(WebSocket示例)  如何在阿里云虚拟主机上快速搭建个人网站?  如何选择可靠的免备案建站服务器?  怎样使用JSON进行数据交换_它有什么限制  邀请函制作网站有哪些,有没有做年会邀请函的网站啊?在线制作,模板很多的那种?  如何将凡科建站内容保存为本地文件?  javascript中的数组方法有哪些_如何利用数组方法简化数据处理  新三国志曹操传主线渭水交兵攻略  制作旅游网站html,怎样注册旅游网站?  如何在IIS中新建站点并配置端口与物理路径?  如何在万网开始建站?分步指南解析  夸克浏览器网页跳转延迟怎么办 夸克浏览器跳转优化  昵图网官方站入口 昵图网素材图库官网入口  如何在云服务器上快速搭建个人网站?  大连网站制作公司哪家好一点,大连买房网站哪个好?  如何快速使用云服务器搭建个人网站?  如何在腾讯云免费申请建站?  在centOS 7安装mysql 5.7的详细教程  北京的网站制作公司有哪些,哪个视频网站最好?  Linux系统运维自动化项目教程_Ansible批量管理实战  Laravel Eloquent关联是什么_Laravel模型一对一与一对多关系精讲  简单实现jsp分页  Laravel如何获取当前登录用户信息_Laravel Auth门面使用与Session用户读取【技巧】  Windows驱动无法加载错误解决方法_驱动签名验证失败处理步骤  Laravel如何实现多对多模型关联?(Eloquent教程)  如何安全更换建站之星模板并保留数据?  Laravel怎么为数据库表字段添加索引以优化查询  长沙做网站要多少钱,长沙国安网络怎么样?  Laravel Blade组件怎么用_Laravel可复用视图组件的创建与使用  大同网页,大同瑞慈医院官网?  Laravel如何使用Service Provider注册服务_Laravel服务提供者配置与加载  EditPlus中的正则表达式实战(6)  Laravel如何实现API版本控制_Laravel版本化API设计方案  Bootstrap整体框架之JavaScript插件架构  百度浏览器如何管理插件 百度浏览器插件管理方法  JS弹性运动实现方法分析  Laravel怎么使用Blade模板引擎_Laravel模板继承与Component组件复用【手册】  高性价比服务器租赁——企业级配置与24小时运维服务  Laravel Eloquent模型如何创建_Laravel ORM基础之Model创建与使用教程