C++中的std::shared_from_this有什么用?C++安全获取this的shared_ptr【智能指针】
技术百科
穿越時空
发布时间:2026-01-01
浏览: 次 std::shared_from_this的作用是让已由std::shared_ptr管理的对象安全生成指向自身的另一个std::shared_ptr;必须公有继承std::enable_shared_from_this,且仅在对象已被shared_ptr拥有后(不可在构造函数中)调用,否则抛std::bad_weak_ptr。
std::shared_from_this 的作用是:让一个已由 std::shared_ptr 管理的对象,能安全地生成指向自身的另一个 std::shared_ptr,避免重复管理或悬空指针。
为什么不能直接用 new 构造 shared_ptr?
如果对象已经由某个 shared_ptr 管理(比如被外部创建并传入),你再用 new 或 make_shared 包装 this,会导致两个独立的控制块,引用计数互不感知——析构两次、内存崩溃、UB(未定义行为)。
正确做法是:对象必须继承自 std::enable_shared_from_this,然后调用 shared_from_this()。
怎么安全使用 shared_from_this?
- 类需公有继承
st
d::enable_shared_from_this - 只能在对象已被
shared_ptr拥有时调用(即:该对象必须是通过make_shared或shared_ptr构造出来的) - 不能在构造函数里调用 —— 此时控制块还没完全建立,会抛
std::bad_weak_ptr - 推荐在成员函数中使用,比如回调注册、异步任务传递自身等场景
典型使用场景举例
比如一个网络连接类需要把自身传给异步读取回调:
class Connection : public std::enable_shared_from_this{ public: void start_read() { auto self = shared_from_this(); // 安全获取自身 shared_ptr socket_.async_read_some(buffer_, [self](auto ec, size_t n) { self->on_read(ec, n); // 即使 Connection 已被释放,self 也能保活 }); } private: tcp::socket socket_; void on_read(std::error_code, size_t) { /* ... */ } };
这样即使外层的 shared_ptr 提前释放,回调里的 self 仍能保证对象存活到回调执行完。
常见错误和注意事项
- 忘记继承
enable_shared_from_this→ 编译失败(shared_from_this未定义) - 对象不是由
shared_ptr创建(如栈对象或裸指针 new)→ 运行时抛std::bad_weak_ptr - 想在构造函数中“提前保存自己” → 不行,控制块尚未就绪;可改用延迟初始化(如第一次调用时 lazy-init)
- 多继承时注意模板参数要写对类型,别写成派生类名以外的别名
基本上就这些。shared_from_this 不复杂,但容易忽略前提条件,用对了才能真正安全。
# 能在
# 也能
# 可在
# 已被
# 是由
# 还没
# 两次
# 再用
# 对象
# c++
# 指针
# 构造函数
# 为什么
# 栈
# 异步
# red
# this
# 回调
# 成员函数
# 继承
# 空指针
# 异步任务
# 公有继承
# 多继承
# 已由
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- Mac如何将HEIC图片格式转为JPG_Mac批量
- Python数据挖掘进阶教程_分类回归与聚类案例解
- Win10如何优化内存使用_Win10内存优化技巧
- Win11如何更改用户账户文件夹名称 Win11修
- php订单日志怎么记录物流_php记录订单物流变更
- Python音视频处理高级项目教程_FFmpegP
- Python装饰器设计思路_功能增强机制说明【指导
- VSC怎么创建PHP项目_从零开始搭建项目的步骤【
- Win11右键反应慢怎么办 Win11优化右键菜单
- Mac电脑如何恢复出厂设置_Mac抹掉数据并重装系
- Win11如何更新显卡驱动 Win11检查和安装设
- php怎么操作Redis_Redis扩展连接与基本
- Windows10如何查看保存的WiFi密码_Wi
- Flask 表单数据通过 SMTP 发送邮件的完整
- Win11时间不对怎么同步_Win11自动校准互联
- 如何在 Go 结构体中正确初始化 map 字段
- 如何在JavaScript中动态拼接PHP的bas
- Windows10系统怎么查看防火墙状态_Win1
- Win11开机Logo怎么换_Win11自定义启动
- Win10如何卸载Skype_Win10卸载Sky
- C++如何使用std::optional?(处理可
- c++怎么实现大文件的分块读写_c++ 文件指针s
- Win10电脑怎么设置网络名称_Windows10
- Win11怎样安装网易云音乐_Win11安装网易云
- 如何在JavaScript中动态拼接PHP的bas
- php怎么连接数据库_MySQL数据库连接的基础代
- php控制舵机角度怎么调_php发送pwm信号控制
- c++输入输出流 c++ cin与cout格式化输
- PHP的Workerman对架构扩展有啥帮助_应用
- php接口返回数据乱码怎么办_php接口调试编码问
- 为什么本地php环境运行php脚本卡顿_php执行
- LINUX怎么查看进程_LINUX ps命令查看运
- 如何高效识别并拦截拼接式恶意域名 spam
- C#如何序列化对象为XML XmlSerializ
- C#怎么创建控制台应用 C# Console Ap
- VSC怎样在Linux运行PHP_Ubuntu系统
- 如何在Golang中实现文件下载_Golang文件
- Windows如何拦截2345弹窗广告_Windo
- 如何使用Golang实现文件加密_Golang c
- Windows笔记本无法进入睡眠模式怎么办?(电源
- Windows10如何重置此电脑_Windows1
- Python解释执行模型_字节码流程说明【指导】
- Windows服务持续崩溃怎样修复_系统服务保护机
- mac怎么打开终端_MAC终端Terminal使用
- php485支持哪些操作系统_php485跨系统支
- Win11怎么关闭应用权限_Windows11相机
- 本地php环境打开php文件直接下载_浏览器解析p
- php转mp4怎么保留字幕_php处理带字幕视频转
- Python函数缓存机制_lru_cache解析【
- Windows10如何更改系统字体大小_Win10

d::enable_shared_from_this
QQ客服