asio中socket通常的用法如
如果此时外界调用了Session::Close又删除了该对象,已经由async_read注册的HandleRead并不会被立即取消,而是会再次被调用,并把error_code设置成operation_aborted。那么在HandleRead里的delete this就会重复删除对象造成崩溃。
如果把HandleRead改成
static void HandleRead(shared_ptr<Session> session, ...)
或者weak_ptr然后检测是否已经expired,可以解决问题,但是由于智能指针占用空间比原生指针大,会在Read()中的boost::bind一句里造成动态内存分配(见此文),造成效率急剧下降。
最终的解决方案:手动去记录有多少异步操作被注册,关闭socket时只有当计数减少到0才销毁对象。